home *** CD-ROM | disk | FTP | other *** search
/ The CICA Windows Explosion! / The CICA Windows Explosion! - Disc 2.iso / nt / emacssrc.zip / EMACSSRC.TAR / emacs-19.17 / src / window.c < prev    next >
C/C++ Source or Header  |  1993-10-07  |  94KB  |  3,080 lines

  1. /* Window creation, deletion and examination for GNU Emacs.
  2.    Does not include redisplay.
  3.    Copyright (C) 1985, 1986, 1987, 1993 Free Software Foundation, Inc.
  4.  
  5. This file is part of GNU Emacs.
  6.  
  7. GNU Emacs is free software; you can redistribute it and/or modify
  8. it under the terms of the GNU General Public License as published by
  9. the Free Software Foundation; either version 2, or (at your option)
  10. any later version.
  11.  
  12. GNU Emacs is distributed in the hope that it will be useful,
  13. but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  15. GNU General Public License for more details.
  16.  
  17. You should have received a copy of the GNU General Public License
  18. along with GNU Emacs; see the file COPYING.  If not, write to
  19. the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */
  20.  
  21. #include "config.h"
  22. #include "lisp.h"
  23. #include "buffer.h"
  24. #include "frame.h"
  25. #include "window.h"
  26. #include "commands.h"
  27. #include "indent.h"
  28. #include "termchar.h"
  29. #include "disptab.h"
  30. #include "keyboard.h"
  31.  
  32. #ifdef STDC_HEADERS
  33. #include <stdlib.h>
  34. #endif
  35. #include "window_p.h"
  36. #include "marker_p.h"
  37. #include "intervals_p.h"
  38. #include "dispnew_d.h"
  39. static struct window * decode_window _P_((register Lisp_Object window));
  40. static int coordinates_in_window _P_((register struct window *w,
  41.                                       register int *x, register int *y));
  42. static _VOID_ unshow_buffer _P_((register struct window *w));
  43. static _VOID_ replace_window _P_((Lisp_Object old, Lisp_Object replacement));
  44. static Lisp_Object window_loop _P_((enum window_loop type,
  45.                                     register Lisp_Object obj, int mini,
  46.                                     register Lisp_Object frames));
  47. static void check_min_window_sizes _P_((void));
  48. static _VOID_ make_dummy_parent _P_((Lisp_Object window));
  49. static void scroll_command _P_((register Lisp_Object n, int direction));
  50. static int count_windows _P_((register struct window *window));
  51. static int save_window_save _P_((Lisp_Object window,
  52.                                  struct Lisp_Vector *vector, int i));
  53.  
  54. Lisp_Object Qwindowp, Qwindow_live_p;
  55.  
  56. /* This is the window in which the terminal's cursor should
  57.    be left when nothing is being done with it.  This must
  58.    always be a leaf window, and its buffer is selected by
  59.    the top level editing loop at the end of each command.
  60.  
  61.    This value is always the same as
  62.    FRAME_SELECTED_WINDOW (selected_frame).  */
  63.  
  64. Lisp_Object selected_window;
  65.  
  66. /* The minibuffer window of the selected frame.
  67.    Note that you cannot test for minibufferness of an arbitrary window
  68.    by comparing against this; but you can test for minibufferness of
  69.    the selected window.  */
  70. Lisp_Object minibuf_window;
  71.  
  72. /* Non-nil means it is the window for C-M-v to scroll
  73.    when the minibuffer is selected.  */
  74. Lisp_Object Vminibuf_scroll_window;
  75.  
  76. /* Non-nil means this is the buffer whose window C-M-v should scroll.  */
  77. Lisp_Object Vother_window_scroll_buffer;
  78.  
  79. /* Non-nil means it's function to call to display temp buffers.  */
  80. Lisp_Object Vtemp_buffer_show_function;
  81.  
  82. /* If a window gets smaller than either of these, it is removed. */
  83. int window_min_height;
  84. int window_min_width;
  85.  
  86. /* Nonzero implies Fdisplay_buffer should create windows. */
  87. int pop_up_windows;
  88.  
  89. /* Nonzero implies make new frames for Fdisplay_buffer.  */
  90. int pop_up_frames;
  91.  
  92. /* Non-nil means use this function instead of default */
  93. Lisp_Object Vpop_up_frame_function;
  94.  
  95. /* Function to call to handle Fdisplay_buffer.  */
  96. Lisp_Object Vdisplay_buffer_function;
  97.  
  98. /* Fdisplay_buffer always splits the largest window 
  99.    if that window is more than this high.  */
  100. int split_height_threshold;
  101.  
  102. /* Number of lines of continuity in scrolling by screenfuls.  */
  103. int next_screen_context_lines;
  104.  
  105. /* Incremented for each window created.  */
  106. static int sequence_number;
  107.  
  108. #ifndef min
  109. #define min(a, b) ((a) < (b) ? (a) : (b))
  110. #endif
  111.  
  112.  
  113. DEFUN ("windowp", Fwindowp, Swindowp, 1, 1, 0,
  114.   "Returns t if OBJ is a window.")
  115.   (obj)
  116.      Lisp_Object obj;
  117. {
  118.   return XTYPE (obj) == Lisp_Window ? Qt : Qnil;
  119. }
  120.  
  121. DEFUN ("window-live-p", Fwindow_live_p, Swindow_live_p, 1, 1, 0,
  122.   "Returns t if OBJ is a window which is currently visible.")
  123.      (obj)
  124.      Lisp_Object obj;
  125. {
  126.   return ((XTYPE (obj) == Lisp_Window
  127.        && ! NILP (XWINDOW (obj)->buffer))
  128.       ? Qt : Qnil);
  129. }
  130.  
  131. Lisp_Object
  132. make_window ()
  133. {
  134.   register Lisp_Object val;
  135.   register struct window *p;
  136.  
  137.   /* Add sizeof (Lisp_Object) here because sizeof (struct Lisp_Vector)
  138.      includes the first element.  */
  139.   val = Fmake_vector (
  140.     make_number ((sizeof (struct window) - sizeof (struct Lisp_Vector)
  141.           + sizeof (Lisp_Object))
  142.          / sizeof (Lisp_Object)),
  143.     Qnil);
  144.   XSETTYPE (val, Lisp_Window);
  145.   p = XWINDOW (val);
  146.   XFASTINT (p->sequence_number) = ++sequence_number;
  147.   XFASTINT (p->left) = XFASTINT (p->top)
  148.     = XFASTINT (p->height) = XFASTINT (p->width)
  149.       = XFASTINT (p->hscroll) = 0;
  150.   XFASTINT (p->last_point_x) = XFASTINT (p->last_point_y) = 0;
  151.   p->start = Fmake_marker ();
  152.   p->pointm = Fmake_marker ();
  153.   XFASTINT (p->use_time) = 0;
  154.   p->frame = Qnil;
  155.   p->display_table = Qnil;
  156.   p->dedicated = Qnil;
  157.   return val;
  158. }
  159.  
  160. DEFUN ("selected-window", Fselected_window, Sselected_window, 0, 0, 0,
  161.   "Return the window that the cursor now appears in and commands apply to.")
  162.   ()
  163. {
  164.   return selected_window;
  165. }
  166.  
  167. DEFUN ("minibuffer-window", Fminibuffer_window, Sminibuffer_window, 0, 1, 0,
  168.   "Return the window used now for minibuffers.\n\
  169. If the optional argument FRAME is specified, return the minibuffer window\n\
  170. used by that frame.")
  171.   (frame)
  172.     Lisp_Object frame;
  173. {
  174. #ifdef MULTI_FRAME
  175.   if (NILP (frame))
  176.     XSET (frame, Lisp_Frame, selected_frame);
  177.   else
  178.     CHECK_LIVE_FRAME (frame, 0);
  179. #endif
  180.  
  181.   return FRAME_MINIBUF_WINDOW (XFRAME (frame));
  182. }
  183.  
  184. DEFUN ("window-minibuffer-p", Fwindow_minibuffer_p, Swindow_minibuffer_p, 0, 1, 0,
  185.   "Returns non-nil if WINDOW is a minibuffer window.")
  186.   (window)
  187.      Lisp_Object window;
  188. {
  189.   struct window *w = decode_window (window);
  190.   return (MINI_WINDOW_P (w) ? Qt : Qnil);
  191. }
  192.  
  193. DEFUN ("pos-visible-in-window-p", Fpos_visible_in_window_p,
  194.   Spos_visible_in_window_p, 0, 2, 0,
  195.   "Return t if position POS is currently on the frame in WINDOW.\n\
  196. Returns nil if that position is scrolled vertically out of view.\n\
  197. POS defaults to point; WINDOW, to the selected window.")
  198.   (pos, window)
  199.      Lisp_Object pos, window;
  200. {
  201.   register struct window *w;
  202.   register int top;
  203.   register int height;
  204.   register int posint;
  205.   register struct buffer *buf;
  206.   struct position posval;
  207.  
  208.   if (NILP (pos))
  209.     posint = point;
  210.   else
  211.     {
  212.       CHECK_NUMBER_COERCE_MARKER (pos, 0);
  213.       posint = XINT (pos);
  214.     }
  215.  
  216.   w = decode_window (window);
  217.   top = marker_position (w->start);
  218.  
  219.   if (posint < top)
  220.     return Qnil;
  221.  
  222.   height = XFASTINT (w->height) - ! MINI_WINDOW_P (w);
  223.  
  224.   buf = XBUFFER (w->buffer);
  225.   if (XFASTINT (w->last_modified) >= BUF_MODIFF (buf))
  226.     {
  227.       /* If frame is up to date,
  228.      use the info recorded about how much text fit on it. */
  229.       if (posint < BUF_Z (buf) - XFASTINT (w->window_end_pos)
  230.       || (XFASTINT (w->window_end_vpos) < height))
  231.     return Qt;
  232.       return Qnil;
  233.     }
  234.   else
  235.     {
  236.       if (posint > BUF_Z (buf))
  237.     return Qnil;
  238.  
  239.       /* If that info is not correct, calculate afresh */
  240.       posval = *compute_motion (top, 0, 0, posint, height, 0,
  241.                 window_internal_width (w) - 1,
  242.                 XINT (w->hscroll), 0);
  243.  
  244.       return posval.vpos < height ? Qt : Qnil;
  245.     }
  246. }
  247.  
  248. static struct window *
  249. decode_window (window)
  250.      register Lisp_Object window;
  251. {
  252.   if (NILP (window))
  253.     return XWINDOW (selected_window);
  254.  
  255.   CHECK_LIVE_WINDOW (window, 0);
  256.   return XWINDOW (window);
  257. }
  258.  
  259. DEFUN ("window-buffer", Fwindow_buffer, Swindow_buffer, 0, 1, 0,
  260.   "Return the buffer that WINDOW is displaying.")
  261.   (window)
  262.      Lisp_Object window;
  263. {
  264.   return decode_window (window)->buffer;
  265. }
  266.  
  267. DEFUN ("window-height", Fwindow_height, Swindow_height, 0, 1, 0,
  268.   "Return the number of lines in WINDOW (including its mode line).")
  269.   (window)
  270.      Lisp_Object window;
  271. {
  272.   return decode_window (window)->height;
  273. }
  274.  
  275. DEFUN ("window-width", Fwindow_width, Swindow_width, 0, 1, 0,
  276.   "Return the number of columns in WINDOW.")
  277.   (window)
  278.      Lisp_Object window;
  279. {
  280.   register struct window *w = decode_window (window);
  281.   register int width = XFASTINT (w->width);
  282.  
  283.   return make_number (window_internal_width (w));
  284. }
  285.  
  286. DEFUN ("window-hscroll", Fwindow_hscroll, Swindow_hscroll, 0, 1, 0,
  287.   "Return the number of columns by which WINDOW is scrolled from left margin.")
  288.   (window)
  289.      Lisp_Object window;
  290. {
  291.   return decode_window (window)->hscroll;
  292. }
  293.  
  294. DEFUN ("set-window-hscroll", Fset_window_hscroll, Sset_window_hscroll, 2, 2, 0,
  295.   "Set number of columns WINDOW is scrolled from left margin to NCOL.\n\
  296. NCOL should be zero or positive.")
  297.   (window, ncol)
  298.      register Lisp_Object window, ncol;
  299. {
  300.   register struct window *w;
  301.  
  302.   CHECK_NUMBER (ncol, 1);
  303.   if (XINT (ncol) < 0) XFASTINT (ncol) = 0;
  304.   if (XFASTINT (ncol) >= (1 << (SHORTBITS - 1)))
  305.     args_out_of_range (ncol, Qnil);
  306.   w = decode_window (window);
  307.   if (XINT (w->hscroll) != XINT (ncol))
  308.     clip_changed = 1;        /* Prevent redisplay shortcuts */
  309.   w->hscroll = ncol;
  310.   return ncol;
  311. }
  312.  
  313. DEFUN ("window-edges", Fwindow_edges, Swindow_edges, 0, 1, 0,
  314.   "Return a list of the edge coordinates of WINDOW.\n\
  315. \(LEFT TOP RIGHT BOTTOM), all relative to 0, 0 at top left corner of frame.\n\
  316. RIGHT is one more than the rightmost column used by WINDOW,\n\
  317. and BOTTOM is one more than the bottommost row used by WINDOW\n\
  318.  and its mode-line.")
  319.   (window)
  320.      Lisp_Object window;
  321. {
  322.   register struct window *w = decode_window (window);
  323.  
  324.   return Fcons (w->left, Fcons (w->top,
  325.            Fcons (make_number (XFASTINT (w->left) + XFASTINT (w->width)),
  326.           Fcons (make_number (XFASTINT (w->top)
  327.                       + XFASTINT (w->height)),
  328.              Qnil))));
  329. }
  330.  
  331. /* Test if the character at column *x, row *y is within window *w.
  332.    If it is not, return 0;
  333.    if it is in the window's text area,
  334.       set *x and *y to its location relative to the upper left corner
  335.          of the window, and
  336.       return 1;
  337.    if it is on the window's modeline, return 2;
  338.    if it is on the border between the window and its right sibling,
  339.       return 3.  */
  340. static int
  341. coordinates_in_window (w, x, y)
  342.      register struct window *w;
  343.      register int *x, *y;
  344. {
  345.   register int left = XINT (w->left);
  346.   register int width = XINT (w->width);
  347.   register int window_height = XINT (w->height);
  348.   register int top = XFASTINT (w->top);
  349.  
  350.   if (   *x < left || *x >= left + width
  351.       || *y < top  || *y >= top  + window_height)
  352.     return 0;
  353.  
  354.   /* Is the character is the mode line?  */
  355.   if (*y == top + window_height - 1
  356.       && ! MINI_WINDOW_P (w))
  357.     return 2;
  358.   
  359.   /* Is the character in the right border?  */
  360.   if (*x == left + width - 1
  361.       && left + width != FRAME_WIDTH (XFRAME (w->frame)))
  362.     return 3;
  363.  
  364.   *x -= left;
  365.   *y -= top;
  366.   return 1;
  367. }
  368.  
  369. DEFUN ("coordinates-in-window-p", Fcoordinates_in_window_p,
  370.   Scoordinates_in_window_p, 2, 2, 0,
  371.   "Return non-nil if COORDINATES are in WINDOW.\n\
  372. COORDINATES is a cons of the form (X . Y), X and Y being distances\n\
  373. measured in characters from the upper-left corner of the frame.\n\
  374. (0 .  0) denotes the character in the upper left corner of the\n\
  375. frame.\n\
  376. If COORDINATES are in the text portion of WINDOW,\n\
  377.    the coordinates relative to the window are returned.\n\
  378. If they are in the mode line of WINDOW, `mode-line' is returned.\n\
  379. If they are on the border between WINDOW and its right sibling,\n\
  380.    `vertical-line' is returned.")
  381.   (coordinates, window)
  382.      register Lisp_Object coordinates, window;
  383. {
  384.   int x, y;
  385.  
  386.   CHECK_LIVE_WINDOW (window, 0);
  387.   CHECK_CONS (coordinates, 1);
  388.   x = XINT (Fcar (coordinates));
  389.   y = XINT (Fcdr (coordinates));
  390.  
  391.   switch (coordinates_in_window (XWINDOW (window), &x, &y))
  392.     {
  393.     case 0:            /* NOT in window at all. */
  394.       return Qnil;
  395.  
  396.     case 1:            /* In text part of window. */
  397.       return Fcons (x, y);
  398.  
  399.     case 2:            /* In mode line of window. */
  400.       return Qmode_line;
  401.       
  402.     case 3:            /* On right border of window.  */
  403.       return Qvertical_line;
  404.  
  405.     default:
  406.       abort ();
  407.     }
  408. }
  409.  
  410. /* Find the window containing column x, row y, and return it as a
  411.    Lisp_Object.  If x, y is on the window's modeline, set *part
  412.    to 1; if it is on the separating line between the window and its
  413.    right sibling, set it to 2; otherwise set it to 0.  If there is no
  414.    window under x, y return nil and leave *part unmodified.  */
  415. Lisp_Object
  416. window_from_coordinates (frame, x, y, part)
  417.      FRAME_PTR frame;
  418.      int x, y;
  419.      int *part;
  420. {
  421.   register Lisp_Object tem, first;
  422.  
  423.   tem = first = FRAME_SELECTED_WINDOW (frame);
  424.  
  425.   do
  426.     {
  427.       int found = coordinates_in_window (XWINDOW (tem), &x, &y);
  428.  
  429.       if (found)
  430.     {
  431.       *part = found - 1;
  432.       return tem;
  433.     }
  434.  
  435.       tem = Fnext_window (tem, Qt, Qlambda);
  436.     }
  437.   while (! EQ (tem, first));
  438.   
  439.   return Qnil;
  440. }
  441.  
  442. DEFUN ("window-at", Fwindow_at, Swindow_at, 2, 3, 0,
  443.   "Return window containing coordinates X and Y on FRAME.\n\
  444. If omitted, FRAME defaults to the currently selected frame.\n\
  445. The top left corner of the frame is considered to be row 0,\n\
  446. column 0.")
  447.   (x, y, frame)
  448.       Lisp_Object x, y, frame;
  449. {
  450.   int part;
  451.  
  452. #ifdef MULTI_FRAME
  453.   if (NILP (frame))
  454.     XSET (frame, Lisp_Frame, selected_frame);
  455.   else
  456.     CHECK_LIVE_FRAME (frame, 2);
  457. #endif
  458.   CHECK_NUMBER (x, 0);
  459.   CHECK_NUMBER (y, 1);
  460.  
  461.   return window_from_coordinates (XFRAME (frame),
  462.                   XINT (x), XINT (y),
  463.                   &part);
  464. }
  465.  
  466. DEFUN ("window-point", Fwindow_point, Swindow_point, 0, 1, 0,
  467.   "Return current value of point in WINDOW.\n\
  468. For a nonselected window, this is the value point would have\n\
  469. if that window were selected.\n\
  470. \n\
  471. Note that, when WINDOW is the selected window and its buffer\n\
  472. is also currently selected, the value returned is the same as (point).\n\
  473. It would be more strictly correct to return the `top-level' value\n\
  474. of point, outside of any save-excursion forms.\n\
  475. But that is hard to define.")
  476.   (window)
  477.      Lisp_Object window;
  478. {
  479.   register struct window *w = decode_window (window);
  480.  
  481.   if (w == XWINDOW (selected_window)
  482.       && current_buffer == XBUFFER (w->buffer))
  483.     return Fpoint ();
  484.   return Fmarker_position (w->pointm);
  485. }
  486.  
  487. DEFUN ("window-start", Fwindow_start, Swindow_start, 0, 1, 0,
  488.   "Return position at which display currently starts in WINDOW.")
  489.   (window)
  490.      Lisp_Object window;
  491. {
  492.   return Fmarker_position (decode_window (window)->start);
  493. }
  494.  
  495. DEFUN ("window-end", Fwindow_end, Swindow_end, 0, 1, 0,
  496.   "Return position at which display currently ends in WINDOW.")
  497.   (window)
  498.      Lisp_Object window;
  499. {
  500.   Lisp_Object value;
  501.   struct window *w = decode_window (window);
  502.   
  503.   XSET (value, Lisp_Int,
  504.     BUF_Z (current_buffer) - XFASTINT (w->window_end_pos));
  505.  
  506.   return value;
  507. }
  508.  
  509. DEFUN ("set-window-point", Fset_window_point, Sset_window_point, 2, 2, 0,
  510.   "Make point value in WINDOW be at position POS in WINDOW's buffer.")
  511.   (window, pos)
  512.      Lisp_Object window, pos;
  513. {
  514.   register struct window *w = decode_window (window);
  515.  
  516.   CHECK_NUMBER_COERCE_MARKER (pos, 1);
  517.   if (w == XWINDOW (selected_window))
  518.     Fgoto_char (pos);
  519.   else
  520.     set_marker_restricted (w->pointm, pos, w->buffer);
  521.  
  522.   return pos;
  523. }
  524.  
  525. DEFUN ("set-window-start", Fset_window_start, Sset_window_start, 2, 3, 0,
  526.   "Make display in WINDOW start at position POS in WINDOW's buffer.\n\
  527. Optional third arg NOFORCE non-nil inhibits next redisplay\n\
  528. from overriding motion of point in order to display at this exact start.")
  529.   (window, pos, noforce)
  530.      Lisp_Object window, pos, noforce;
  531. {
  532.   register struct window *w = decode_window (window);
  533.  
  534.   CHECK_NUMBER_COERCE_MARKER (pos, 1);
  535.   set_marker_restricted (w->start, pos, w->buffer);
  536.   /* this is not right, but much easier than doing what is right. */
  537.   w->start_at_line_beg = Qnil;
  538.   if (NILP (noforce))
  539.     w->force_start = Qt;
  540.   w->update_mode_line = Qt;
  541.   XFASTINT (w->last_modified) = 0;
  542.   if (!EQ (window, selected_window))
  543.     windows_or_buffers_changed++;
  544.   return pos;
  545. }
  546.  
  547. DEFUN ("window-dedicated-p", Fwindow_dedicated_p, Swindow_dedicated_p,
  548.        1, 1, 0,
  549.   "Return WINDOW's dedicated object, usually t or nil.\n\
  550. See also `set-window-dedicated-p'.")
  551.   (window)
  552.      Lisp_Object window;
  553. {
  554.   return decode_window (window)->dedicated;
  555. }
  556.  
  557. DEFUN ("set-window-dedicated-p", Fset_window_dedicated_p,
  558.        Sset_window_dedicated_p, 2, 2, 0,
  559.   "Control whether WINDOW is dedicated to the buffer it displays.\n\
  560. If it is dedicated, Emacs will not automatically change\n\
  561. which buffer appears in it.\n\
  562. The second argument is the new value for the dedication flag;\n\
  563. non-nil means yes.")
  564.   (window, arg)
  565.        Lisp_Object window, arg;
  566. {
  567.   register struct window *w = decode_window (window);
  568.  
  569.   if (NILP (arg))
  570.     w->dedicated = Qnil;
  571.   else
  572.     w->dedicated = Qt;
  573.  
  574.   return w->dedicated;
  575. }
  576.  
  577. DEFUN ("window-display-table", Fwindow_display_table, Swindow_display_table,
  578.        0, 1, 0,
  579.   "Return the display-table that WINDOW is using.")
  580.   (window)
  581.      Lisp_Object window;
  582. {
  583.   return decode_window (window)->display_table;
  584. }
  585.  
  586. /* Get the display table for use currently on window W.
  587.    This is either W's display table or W's buffer's display table.
  588.    Ignore the specified tables if they are not valid;
  589.    if no valid table is specified, return 0.  */
  590.  
  591. struct Lisp_Vector *
  592. window_display_table (w)
  593.      struct window *w;
  594. {
  595.   Lisp_Object tem;
  596.   tem = w->display_table;
  597.   if (XTYPE (tem) == Lisp_Vector && XVECTOR (tem)->size == DISP_TABLE_SIZE)
  598.     return XVECTOR (tem);
  599.   tem = XBUFFER (w->buffer)->display_table;
  600.   if (XTYPE (tem) == Lisp_Vector && XVECTOR (tem)->size == DISP_TABLE_SIZE)
  601.     return XVECTOR (tem);
  602.   tem = Vstandard_display_table;
  603.   if (XTYPE (tem) == Lisp_Vector && XVECTOR (tem)->size == DISP_TABLE_SIZE)
  604.     return XVECTOR (tem);
  605.   return 0;
  606. }
  607.  
  608. DEFUN ("set-window-display-table", Fset_window_display_table, Sset_window_display_table, 2, 2, 0,
  609.   "Set WINDOW's display-table to TABLE.")
  610.   (window, table)
  611.      register Lisp_Object window, table;
  612. {
  613.   register struct window *w;
  614.  
  615.   w = decode_window (window);
  616.   w->display_table = table;
  617.   return table;
  618. }
  619.  
  620. /* Record info on buffer window w is displaying
  621.    when it is about to cease to display that buffer.  */
  622. static _VOID_
  623. unshow_buffer (w)
  624.      register struct window *w;
  625. {
  626.   Lisp_Object buf = w->buffer;
  627.  
  628.   if (XBUFFER (buf) != XMARKER (w->pointm)->buffer)
  629.     abort ();
  630.  
  631.   if (w == XWINDOW (selected_window)
  632.       || ! EQ (buf, XWINDOW (selected_window)->buffer))
  633.     /* Do this except when the selected window's buffer
  634.        is being removed from some other window.  */
  635.     XBUFFER (buf)->last_window_start = marker_position (w->start);
  636.  
  637.   /* Point in the selected window's buffer
  638.      is actually stored in that buffer, and the window's pointm isn't used.
  639.      So don't clobber point in that buffer.  */
  640.   if (! EQ (buf, XWINDOW (selected_window)->buffer))
  641.     BUF_PT (XBUFFER (buf))
  642.       = clip_to_bounds (BUF_BEGV (XBUFFER (buf)),
  643.             marker_position (w->pointm),
  644.             BUF_ZV (XBUFFER (buf)));
  645. }
  646.  
  647. /* Put replacement into the window structure in place of old. */
  648. static _VOID_
  649. replace_window (old, replacement)
  650.      Lisp_Object old, replacement;
  651. {
  652.   register Lisp_Object tem;
  653.   register struct window *o = XWINDOW (old), *p = XWINDOW (replacement);
  654.  
  655.   /* If OLD is its frame's root_window, then replacement is the new
  656.      root_window for that frame.  */
  657.  
  658.   if (EQ (old, FRAME_ROOT_WINDOW (XFRAME (o->frame))))
  659.     FRAME_ROOT_WINDOW (XFRAME (o->frame)) = replacement;
  660.  
  661.   p->left = o->left;
  662.   p->top = o->top;
  663.   p->width = o->width;
  664.   p->height = o->height;
  665.  
  666.   p->next = tem = o->next;
  667.   if (!NILP (tem))
  668.     XWINDOW (tem)->prev = replacement;
  669.  
  670.   p->prev = tem = o->prev;
  671.   if (!NILP (tem))
  672.     XWINDOW (tem)->next = replacement;
  673.  
  674.   p->parent = tem = o->parent;
  675.   if (!NILP (tem))
  676.     {
  677.       if (EQ (XWINDOW (tem)->vchild, old))
  678.     XWINDOW (tem)->vchild = replacement;
  679.       if (EQ (XWINDOW (tem)->hchild, old))
  680.     XWINDOW (tem)->hchild = replacement;
  681.     }
  682.  
  683. /*** Here, if replacement is a vertical combination
  684. and so is its new parent, we should make replacement's
  685. children be children of that parent instead.  ***/
  686. }
  687.  
  688. DEFUN ("delete-window", Fdelete_window, Sdelete_window, 0, 1, "",
  689.   "Remove WINDOW from the display.  Default is selected window.")
  690.   (window)
  691.      register Lisp_Object window;
  692. {
  693.   register Lisp_Object tem, parent, sib;
  694.   register struct window *p;
  695.   register struct window *par;
  696.  
  697.   /* Because this function is called by other C code on non-leaf
  698.      windows, the CHECK_LIVE_WINDOW macro would choke inappropriately,
  699.      so we can't decode_window here.  */
  700.   if (NILP (window))
  701.     window = selected_window;
  702.   else
  703.     CHECK_WINDOW (window, 0);
  704.   p = XWINDOW (window);
  705.  
  706.   /* It's okay to delete an already-deleted window.  */
  707.   if (NILP (p->buffer)
  708.       && NILP (p->hchild)
  709.       && NILP (p->vchild))
  710.     return Qnil;
  711.  
  712.   parent = p->parent;
  713.   if (NILP (parent))
  714.     error ("Attempt to delete minibuffer or sole ordinary window");
  715.   par = XWINDOW (parent);
  716.  
  717.   windows_or_buffers_changed++;
  718.  
  719.   /* Are we trying to delete any frame's selected window?  */
  720.   {
  721.     Lisp_Object frame, pwindow;
  722.  
  723.     /* See if the frame's selected window is either WINDOW
  724.        or any subwindow of it, by finding all that window's parents
  725.        and comparing each one with WINDOW.  */
  726.     frame = WINDOW_FRAME (XWINDOW (window));
  727.     pwindow = FRAME_SELECTED_WINDOW (XFRAME (frame));
  728.  
  729.     while (!NILP (pwindow))
  730.       {
  731.     if (EQ (window, pwindow))
  732.       break;
  733.     pwindow = XWINDOW (pwindow)->parent;
  734.       }
  735.  
  736.     if (EQ (window, pwindow))
  737.       {
  738.     Lisp_Object alternative = Fnext_window (window, Qlambda, Qnil);
  739.  
  740.     /* If we're about to delete the selected window on the
  741.        selected frame, then we should use Fselect_window to select
  742.        the new window.  On the other hand, if we're about to
  743.        delete the selected window on any other frame, we shouldn't do
  744.        anything but set the frame's selected_window slot.  */
  745.     if (EQ (window, selected_window))
  746.       Fselect_window (alternative);
  747.     else
  748.       FRAME_SELECTED_WINDOW (XFRAME (frame)) = alternative;
  749.       }
  750.   }
  751.  
  752.   tem = p->buffer;
  753.   /* tem is null for dummy parent windows
  754.      (which have inferiors but not any contents themselves) */
  755.   if (!NILP (tem))
  756.     {
  757.       unshow_buffer (p);
  758.       unchain_marker (p->pointm);
  759.       unchain_marker (p->start);
  760.     }
  761.  
  762.   tem = p->next;
  763.   if (!NILP (tem))
  764.     XWINDOW (tem)->prev = p->prev;
  765.  
  766.   tem = p->prev;
  767.   if (!NILP (tem))
  768.     XWINDOW (tem)->next = p->next;
  769.  
  770.   if (EQ (window, par->hchild))
  771.     par->hchild = p->next;
  772.   if (EQ (window, par->vchild))
  773.     par->vchild = p->next;
  774.  
  775.   /* Find one of our siblings to give our space to.  */
  776.   sib = p->prev;
  777.   if (NILP (sib))
  778.     {
  779.       /* If p gives its space to its next sibling, that sibling needs
  780.      to have its top/left side pulled back to where p's is.
  781.      set_window_{height,width} will re-position the sibling's
  782.      children.  */
  783.       sib = p->next;
  784.       XWINDOW (sib)->top = p->top;
  785.       XWINDOW (sib)->left = p->left;
  786.     }
  787.  
  788.   /* Stretch that sibling.  */
  789.   if (!NILP (par->vchild))
  790.     set_window_height (sib,
  791.                XFASTINT (XWINDOW (sib)->height) + XFASTINT (p->height),
  792.                1);
  793.   if (!NILP (par->hchild))
  794.     set_window_width (sib,
  795.               XFASTINT (XWINDOW (sib)->width) + XFASTINT (p->width),
  796.               1);
  797.  
  798.   /* If parent now has only one child,
  799.      put the child into the parent's place.  */
  800.   tem = par->hchild;
  801.   if (NILP (tem))
  802.     tem = par->vchild;
  803.   if (NILP (XWINDOW (tem)->next))
  804.     replace_window (parent, tem);
  805.  
  806.   /* Since we may be deleting combination windows, we must make sure that
  807.      not only p but all its children have been marked as deleted.  */
  808.   if (! NILP (p->hchild))
  809.     delete_all_subwindows (XWINDOW (p->hchild));
  810.   else if (! NILP (p->vchild))
  811.     delete_all_subwindows (XWINDOW (p->vchild));
  812.  
  813.   /* Mark this window as deleted.  */
  814.   p->buffer = p->hchild = p->vchild = Qnil;
  815.  
  816.   return Qnil;
  817. }
  818.  
  819.  
  820. /* This comment supplies the doc string for `next-window',
  821.    for make-docfile to see.  We cannot put this in the real DEFUN
  822.    due to limits in the Unix cpp.
  823.  
  824. DEFUN ("next-window", Ffoo, Sfoo, 0, 3, 0,
  825.   "Return next window after WINDOW in canonical ordering of windows.\n\
  826. If omitted, WINDOW defaults to the selected window.\n\
  827. \n\
  828. Optional second arg MINIBUF t means count the minibuffer window even\n\
  829. if not active.  MINIBUF nil or omitted means count the minibuffer iff\n\
  830. it is active.  MINIBUF neither t nor nil means not to count the\n\
  831. minibuffer even if it is active.\n\
  832. \n\
  833. Several frames may share a single minibuffer; if the minibuffer\n\
  834. counts, all windows on all frames that share that minibuffer count\n\
  835. too.  This means that next-window may be used to iterate through the\n\
  836. set of windows even when the minibuffer is on another frame.  If the\n\
  837. minibuffer does not count, only windows from WINDOW's frame count.\n\
  838. \n\
  839. Optional third arg ALL-FRAMES t means include windows on all frames.\n\
  840. ALL-FRAMES nil or omitted means cycle within the frames as specified\n\
  841. above.  If neither nil nor t, restrict to WINDOW's frame.\n\
  842. \n\
  843. If you use consistent values for MINIBUF and ALL-FRAMES, you can use\n\
  844. `next-window' to iterate through the entire cycle of acceptable\n\
  845. windows, eventually ending up back at the window you started with.\n\
  846. `previous-window' traverses the same cycle, in the reverse order.")
  847.   (window, minibuf, all_frames) */
  848.  
  849. DEFUN ("next-window", Fnext_window, Snext_window, 0, 3, 0,
  850.        0)
  851.   (window, minibuf, all_frames)
  852.      register Lisp_Object window, minibuf, all_frames;
  853. {
  854.   register Lisp_Object tem;
  855.   Lisp_Object start_window;
  856.  
  857.   if (NILP (window))
  858.     window = selected_window;
  859.   else
  860.     CHECK_LIVE_WINDOW (window, 0);
  861.  
  862.   start_window = window;
  863.  
  864.   /* minibuf == nil may or may not include minibuffers.
  865.      Decide if it does.  */
  866.   if (NILP (minibuf))
  867.     minibuf = (minibuf_level ? Qt : Qlambda);
  868.  
  869.   /* all_frames == nil doesn't specify which frames to include.
  870.      Decide which frames it includes.  */
  871.   if (NILP (all_frames))
  872.     all_frames = (EQ (minibuf, Qt)
  873.            ? (FRAME_MINIBUF_WINDOW
  874.               (XFRAME
  875.                (WINDOW_FRAME
  876.             (XWINDOW (window)))))
  877.            : Qnil);
  878.   else if (! EQ (all_frames, Qt))
  879.     all_frames = Qnil;
  880.   /* Now all_frames is t meaning search all frames,
  881.      nil meaning search just current frame,
  882.      or a window, meaning search the frame that window belongs to.  */
  883.  
  884.   /* Do this loop at least once, to get the next window, and perhaps
  885.      again, if we hit the minibuffer and that is not acceptable.  */
  886.   do
  887.     {
  888.       /* Find a window that actually has a next one.  This loop
  889.      climbs up the tree.  */
  890.       while (tem = XWINDOW (window)->next, NILP (tem))
  891.     if (tem = XWINDOW (window)->parent, !NILP (tem))
  892.       window = tem;
  893.     else
  894.       {
  895.         /* We've reached the end of this frame.
  896.            Which other frames are acceptable?  */
  897.         tem = WINDOW_FRAME (XWINDOW (window));
  898. #ifdef MULTI_FRAME
  899.         if (! NILP (all_frames))
  900.           tem = next_frame (tem, all_frames);
  901. #endif
  902.         tem = FRAME_ROOT_WINDOW (XFRAME (tem));
  903.  
  904.         break;
  905.       }
  906.  
  907.       window = tem;
  908.  
  909.       /* If we're in a combination window, find its first child and
  910.      recurse on that.  Otherwise, we've found the window we want.  */
  911.       while (1)
  912.     {
  913.       if (!NILP (XWINDOW (window)->hchild))
  914.         window = XWINDOW (window)->hchild;
  915.       else if (!NILP (XWINDOW (window)->vchild))
  916.         window = XWINDOW (window)->vchild;
  917.       else break;
  918.     }
  919.     }
  920.   /* Which windows are acceptible?
  921.      Exit the loop and accept this window if
  922.      this isn't a minibuffer window, or
  923.      we're accepting minibuffer windows, or
  924.      we've come all the way around and we're back at the original window.  */
  925.   while (MINI_WINDOW_P (XWINDOW (window))
  926.      && ! EQ (minibuf, Qt)
  927.      && ! EQ (window, start_window));
  928.  
  929.   return window;
  930. }
  931.  
  932. /* This comment supplies the doc string for `previous-window',
  933.    for make-docfile to see.  We cannot put this in the real DEFUN
  934.    due to limits in the Unix cpp.
  935.  
  936. DEFUN ("previous-window", Ffoo, Sfoo, 0, 3, 0,
  937.   "Return the window preceeding WINDOW in canonical ordering of windows.\n\
  938. If omitted, WINDOW defaults to the selected window.\n\
  939. \n\
  940. Optional second arg MINIBUF t means count the minibuffer window even\n\
  941. if not active.  MINIBUF nil or omitted means count the minibuffer iff\n\
  942. it is active.  MINIBUF neither t nor nil means not to count the\n\
  943. minibuffer even if it is active.\n\
  944. \n\
  945. Several frames may share a single minibuffer; if the minibuffer\n\
  946. counts, all windows on all frames that share that minibuffer count\n\
  947. too.  This means that previous-window may be used to iterate through\n\
  948. the set of windows even when the minibuffer is on another frame.  If\n\
  949. the minibuffer does not count, only windows from WINDOW's frame\n\
  950. count.\n\
  951. \n\
  952. Optional third arg ALL-FRAMES t means include windows on all frames.\n\
  953. ALL-FRAMES nil or omitted means cycle within the frames as specified\n\
  954. above.  If neither nil nor t, restrict to WINDOW's frame.\n\
  955. \n\
  956. If you use consistent values for MINIBUF and ALL-FRAMES, you can use\n\
  957. `previous-window' to iterate through the entire cycle of acceptable\n\
  958. windows, eventually ending up back at the window you started with.\n\
  959. `next-window' traverses the same cycle, in the reverse order.")
  960.   (window, minibuf, all_frames)  */
  961.  
  962.  
  963. DEFUN ("previous-window", Fprevious_window, Sprevious_window, 0, 3, 0,
  964.        0)
  965.   (window, minibuf, all_frames)
  966.      register Lisp_Object window, minibuf, all_frames;
  967. {
  968.   register Lisp_Object tem;
  969.   Lisp_Object start_window;
  970.  
  971.   if (NILP (window))
  972.     window = selected_window;
  973.   else
  974.     CHECK_LIVE_WINDOW (window, 0);
  975.  
  976.   start_window = window;
  977.  
  978.   /* minibuf == nil may or may not include minibuffers.
  979.      Decide if it does.  */
  980.   if (NILP (minibuf))
  981.     minibuf = (minibuf_level ? Qt : Qlambda);
  982.  
  983.   /* all_frames == nil doesn't specify which frames to include.
  984.      Decide which frames it includes.  */
  985.   if (NILP (all_frames))
  986.     all_frames = (EQ (minibuf, Qt)
  987.            ? (FRAME_MINIBUF_WINDOW
  988.               (XFRAME
  989.                (WINDOW_FRAME
  990.             (XWINDOW (window)))))
  991.            : Qnil);
  992.   else if (! EQ (all_frames, Qt))
  993.     all_frames = Qnil;
  994.   /* Now all_frames is t meaning search all frames,
  995.      nil meaning search just current frame,
  996.      or a window, meaning search the frame that window belongs to.  */
  997.  
  998.   /* Do this loop at least once, to get the previous window, and perhaps
  999.      again, if we hit the minibuffer and that is not acceptable.  */
  1000.   do
  1001.     {
  1002.       /* Find a window that actually has a previous one.  This loop
  1003.      climbs up the tree.  */
  1004.       while (tem = XWINDOW (window)->prev, NILP (tem))
  1005.     if (tem = XWINDOW (window)->parent, !NILP (tem))
  1006.       window = tem;
  1007.     else
  1008.       {
  1009.         /* We have found the top window on the frame.
  1010.            Which frames are acceptable?  */
  1011.         tem = WINDOW_FRAME (XWINDOW (window));
  1012. #ifdef MULTI_FRAME
  1013.         if (! NILP (all_frames))
  1014.           /* It's actually important that we use prev_frame here,
  1015.          rather than next_frame.  All the windows acceptable
  1016.          according to the given parameters should form a ring;
  1017.          Fnext_window and Fprevious_window should go back and
  1018.          forth around the ring.  If we use next_frame here,
  1019.          then Fnext_window and Fprevious_window take different
  1020.          paths through the set of acceptable windows.
  1021.          window_loop assumes that these `ring' requirement are
  1022.          met.  */
  1023.           tem = prev_frame (tem, all_frames);
  1024. #endif
  1025.         /* If this frame has a minibuffer, find that window first,
  1026.            because it is conceptually the last window in that frame.  */
  1027.         if (FRAME_HAS_MINIBUF_P (XFRAME (tem)))
  1028.           tem = FRAME_MINIBUF_WINDOW (XFRAME (tem));
  1029.         else
  1030.           tem = FRAME_ROOT_WINDOW (XFRAME (tem));
  1031.  
  1032.         break;
  1033.       }
  1034.  
  1035.       window = tem;
  1036.       /* If we're in a combination window, find its last child and
  1037.      recurse on that.  Otherwise, we've found the window we want.  */
  1038.       while (1)
  1039.     {
  1040.       if (!NILP (XWINDOW (window)->hchild))
  1041.         window = XWINDOW (window)->hchild;
  1042.       else if (!NILP (XWINDOW (window)->vchild))
  1043.         window = XWINDOW (window)->vchild;
  1044.       else break;
  1045.       while (tem = XWINDOW (window)->next, !NILP (tem))
  1046.         window = tem;
  1047.     }
  1048.     }
  1049.   /* Which windows are acceptable?
  1050.      Exit the loop and accept this window if
  1051.      this isn't a minibuffer window, or
  1052.      we're accepting minibuffer windows, or
  1053.      we've come all the way around and we're back at the original window.  */
  1054.   while (MINI_WINDOW_P (XWINDOW (window))
  1055.      && !EQ (minibuf, Qt)
  1056.      && !EQ (window, start_window));
  1057.  
  1058.   return window;
  1059. }
  1060.  
  1061. DEFUN ("other-window", Fother_window, Sother_window, 1, 2, "p",
  1062.   "Select the ARG'th different window on this frame.\n\
  1063. All windows on current frame are arranged in a cyclic order.\n\
  1064. This command selects the window ARG steps away in that order.\n\
  1065. A negative ARG moves in the opposite order.  If the optional second\n\
  1066. argument ALL_FRAMES is non-nil, cycle through all frames.")
  1067.   (n, all_frames)
  1068.      register Lisp_Object n, all_frames;
  1069. {
  1070.   register int i;
  1071.   register Lisp_Object w;
  1072.  
  1073.   CHECK_NUMBER (n, 0);
  1074.   w = selected_window;
  1075.   i = XINT (n);
  1076.  
  1077.   while (i > 0)
  1078.     {
  1079.       w = Fnext_window (w, Qnil, all_frames);
  1080.       i--;
  1081.     }
  1082.   while (i < 0)
  1083.     {
  1084.       w = Fprevious_window (w, Qnil, all_frames);
  1085.       i++;
  1086.     }
  1087.   Fselect_window (w);
  1088.   return Qnil;
  1089. }
  1090.  
  1091. /* Look at all windows, performing an operation specified by TYPE
  1092.    with argument OBJ.
  1093.    If FRAMES is Qt, look at all frames;
  1094.                 Qnil, look at just the selected frame;
  1095.             a frame, just look at windows on that frame.
  1096.    If MINI is non-zero, perform the operation on minibuffer windows too.
  1097. */
  1098.  
  1099. enum window_loop
  1100. {
  1101.   WINDOW_LOOP_UNUSED,
  1102.   GET_BUFFER_WINDOW,        /* Arg is buffer */
  1103.   GET_LRU_WINDOW,        /* Arg is t for full-width windows only */
  1104.   DELETE_OTHER_WINDOWS,        /* Arg is window not to delete */
  1105.   DELETE_BUFFER_WINDOWS,    /* Arg is buffer */
  1106.   GET_LARGEST_WINDOW,
  1107.   UNSHOW_BUFFER,        /* Arg is buffer */
  1108. };
  1109.  
  1110. static Lisp_Object
  1111. window_loop (type, obj, mini, frames)
  1112.      enum window_loop type;
  1113.      register Lisp_Object obj, frames;
  1114.      int mini;
  1115. {
  1116.   register Lisp_Object w;
  1117.   register Lisp_Object best_window;
  1118.   register Lisp_Object next_window;
  1119.   register Lisp_Object last_window;
  1120.   FRAME_PTR frame;
  1121.  
  1122. #ifdef MULTI_FRAME
  1123.   /* If we're only looping through windows on a particular frame,
  1124.      frame points to that frame.  If we're looping through windows
  1125.      on all frames, frame is 0.  */
  1126.   if (FRAMEP (frames))
  1127.     frame = XFRAME (frames);
  1128.   else if (NILP (frames))
  1129.     frame = selected_frame;
  1130.   else
  1131.     frame = 0;
  1132. #else
  1133.   frame = 0;
  1134. #endif
  1135.  
  1136.   /* Pick a window to start with.  */
  1137.   if (XTYPE (obj) == Lisp_Window)
  1138.     w = obj;
  1139.   else if (frame)
  1140.     w = FRAME_SELECTED_WINDOW (frame);
  1141.   else
  1142.     w = FRAME_SELECTED_WINDOW (selected_frame);
  1143.  
  1144.   /* Figure out the last window we're going to mess with.  Since
  1145.      Fnext_window, given the same options, is guaranteed to go in a
  1146.      ring, we can just use Fprevious_window to find the last one.
  1147.  
  1148.      We can't just wait until we hit the first window again, because
  1149.      it might be deleted.  */
  1150.  
  1151. #ifdef MULTI_FRAME
  1152.   if (frame)
  1153.     last_window = Fprevious_window (w, (mini ? Qt : Qnil), Qlambda);
  1154.   else
  1155. #endif    /* MULTI_FRAME */
  1156.     /* We know frame is 0, so we're looping through all frames.
  1157.        Or we know this isn't a MULTI_FRAME Emacs, so who cares?  */
  1158.     last_window = Fprevious_window (w, mini ? Qt : Qnil, Qt);
  1159.  
  1160.   best_window = Qnil;
  1161.   for (;;)
  1162.     {
  1163.       FRAME_PTR w_frame = XFRAME (WINDOW_FRAME (XWINDOW (w)));
  1164.  
  1165.       /* Pick the next window now, since some operations will delete
  1166.      the current window.  */
  1167. #ifdef MULTI_FRAME
  1168.       if (frame)
  1169.     next_window = Fnext_window (w, (mini ? Qt : Qnil), Qlambda);
  1170.       else
  1171. #endif                /* MULTI_FRAME */
  1172.     /* We know frame is 0, so we're looping through all frames.
  1173.        Or we know this isn't a MULTI_FRAME Emacs, so who cares?  */
  1174.     next_window = Fnext_window (w, mini ? Qt : Qnil, Qt);
  1175.  
  1176.       if (! MINI_WINDOW_P (XWINDOW (w))
  1177.       || (mini && minibuf_level > 0))
  1178.     switch (type)
  1179.       {
  1180.       case GET_BUFFER_WINDOW:
  1181.         /* Ignore invisible and iconified frames.  */
  1182.         if (! FRAME_VISIBLE_P (w_frame)
  1183.         || FRAME_ICONIFIED_P (w_frame))
  1184.           break;
  1185.         if (XBUFFER (XWINDOW (w)->buffer) == XBUFFER (obj))
  1186.           return w;
  1187.         break;
  1188.  
  1189.       case GET_LRU_WINDOW:
  1190.         /* t as arg means consider only full-width windows */
  1191.         if (!NILP (obj) && XFASTINT (XWINDOW (w)->width)
  1192.         != FRAME_WIDTH (XFRAME (WINDOW_FRAME (XWINDOW (w)))))
  1193.           break;
  1194. #if 0
  1195.         /* Ignore invisible and iconified frames.  */
  1196.         if (! FRAME_VISIBLE_P (XFRAME (WINDOW_FRAME (XWINDOW (w))))
  1197.         || FRAME_ICONIFIED_P (XFRAME (WINDOW_FRAME (XWINDOW (w)))))
  1198.           break;
  1199. #endif
  1200.         /* Ignore dedicated windows and minibuffers.  */
  1201.         if (MINI_WINDOW_P (XWINDOW (w))
  1202.         || !NILP (XWINDOW (w)->dedicated))
  1203.           break;
  1204.         if (NILP (best_window)
  1205.         || (XFASTINT (XWINDOW (best_window)->use_time)
  1206.             > XFASTINT (XWINDOW (w)->use_time)))
  1207.           best_window = w;
  1208.         break;
  1209.  
  1210.       case DELETE_OTHER_WINDOWS:
  1211.         if (XWINDOW (w) != XWINDOW (obj))
  1212.           Fdelete_window (w);
  1213.         break;
  1214.  
  1215.       case DELETE_BUFFER_WINDOWS:
  1216.         if (EQ (XWINDOW (w)->buffer, obj))
  1217.           {
  1218.         /* If we're deleting the buffer displayed in the only window
  1219.            on the frame, find a new buffer to display there.  */
  1220.         if (NILP (XWINDOW (w)->parent))
  1221.           {
  1222.             Lisp_Object new_buffer = Fother_buffer (obj, Qnil);
  1223.             if (NILP (new_buffer))
  1224.               new_buffer
  1225.             = Fget_buffer_create (build_string ("*scratch*"));
  1226.             Fset_window_buffer (w, new_buffer);
  1227.             Fset_buffer (XWINDOW (w)->buffer);
  1228.           }
  1229.         else
  1230.           Fdelete_window (w);
  1231.           }
  1232.         break;
  1233.  
  1234.       case GET_LARGEST_WINDOW:
  1235. #if 0
  1236.         /* Ignore invisible and iconified frames.  */
  1237.         if (! FRAME_VISIBLE_P (XFRAME (WINDOW_FRAME (XWINDOW (w))))
  1238.         || FRAME_ICONIFIED_P (XFRAME (WINDOW_FRAME (XWINDOW (w)))))
  1239.           break;
  1240. #endif
  1241.         /* Ignore dedicated windows and minibuffers.  */
  1242.         if (MINI_WINDOW_P (XWINDOW (w))
  1243.         || !NILP (XWINDOW (w)->dedicated))
  1244.           break;
  1245.         {
  1246.           struct window *best_window_ptr = XWINDOW (best_window);
  1247.           struct window *w_ptr = XWINDOW (w);
  1248.           if (NILP (best_window) ||
  1249.           (XFASTINT (w_ptr->height) * XFASTINT (w_ptr->width))
  1250.           > (XFASTINT (best_window_ptr->height)
  1251.              * XFASTINT (best_window_ptr->width)))
  1252.         best_window = w;
  1253.         }
  1254.         break;
  1255.  
  1256.       case UNSHOW_BUFFER:
  1257.         if (EQ (XWINDOW (w)->buffer, obj))
  1258.           {
  1259.         /* Find another buffer to show in this window.  */
  1260.         Lisp_Object another_buffer = Fother_buffer (obj, Qnil);
  1261.         if (NILP (another_buffer))
  1262.           another_buffer
  1263.             = Fget_buffer_create (build_string ("*scratch*"));
  1264.         Fset_window_buffer (w, another_buffer);
  1265.         if (EQ (w, selected_window))
  1266.           Fset_buffer (XWINDOW (w)->buffer);
  1267.           }
  1268.         break;
  1269.       }
  1270.  
  1271.       if (EQ (w, last_window))
  1272.     break;
  1273.  
  1274.       w = next_window;
  1275.     }
  1276.  
  1277.   return best_window;
  1278. }     
  1279.  
  1280. DEFUN ("get-lru-window", Fget_lru_window, Sget_lru_window, 0, 1, 0,
  1281.   "Return the window least recently selected or used for display.\n\
  1282. If optional argument FRAMES is t, search all frames.  If FRAME is a\n\
  1283. frame, search only that frame.\n")
  1284.   (frames)
  1285.     Lisp_Object frames;
  1286. {
  1287.   register Lisp_Object w;
  1288.   /* First try for a window that is full-width */
  1289.   w = window_loop (GET_LRU_WINDOW, Qt, 0, frames);
  1290.   if (!NILP (w) && !EQ (w, selected_window))
  1291.     return w;
  1292.   /* If none of them, try the rest */
  1293.   return window_loop (GET_LRU_WINDOW, Qnil, 0, frames);
  1294. }
  1295.  
  1296. DEFUN ("get-largest-window", Fget_largest_window, Sget_largest_window, 0, 1, 0,
  1297.   "Return the largest window in area.\n\
  1298. If optional argument FRAMES is t, search all frames.  If FRAME is a\n\
  1299. frame, search only that frame.\n")
  1300.   (frame)
  1301.     Lisp_Object frame;
  1302. {
  1303.   return window_loop (GET_LARGEST_WINDOW, Qnil, 0,
  1304.               frame);
  1305. }
  1306.  
  1307. DEFUN ("get-buffer-window", Fget_buffer_window, Sget_buffer_window, 1, 2, 0,
  1308.   "Return a window currently displaying BUFFER, or nil if none.\n\
  1309. If optional argument FRAME is t, search all visible frames.\n\
  1310. If FRAME is nil, search only the selected frame.\n\
  1311. If FRAME is a frame, search only that frame.\n")
  1312.   (buffer, frame)
  1313.     Lisp_Object buffer, frame;
  1314. {
  1315.   buffer = Fget_buffer (buffer);
  1316.   if (XTYPE (buffer) == Lisp_Buffer)
  1317.     return window_loop (GET_BUFFER_WINDOW, buffer, 1, frame);
  1318.   else
  1319.     return Qnil;
  1320. }
  1321.  
  1322. DEFUN ("delete-other-windows", Fdelete_other_windows, Sdelete_other_windows,
  1323.   0, 1, "",
  1324.   "Make WINDOW (or the selected window) fill its frame.\n\
  1325. Only the frame WINDOW is on is affected.")
  1326.   (window)
  1327.      Lisp_Object window;
  1328. {
  1329.   struct window *w;
  1330.   int opoint = point;
  1331.   struct buffer *obuf = current_buffer;
  1332.   int top;
  1333.  
  1334.   if (NILP (window))
  1335.     window = selected_window;
  1336.   else
  1337.     CHECK_LIVE_WINDOW (window, 0);
  1338.  
  1339.   w = XWINDOW (window);
  1340.   top = XFASTINT (w->top);
  1341.  
  1342.   window_loop (DELETE_OTHER_WINDOWS, window, 0, WINDOW_FRAME (w));
  1343.  
  1344.   Fset_buffer (w->buffer);
  1345.   SET_PT (marker_position (w->start));
  1346.   Frecenter (make_number (top - FRAME_MENU_BAR_LINES (XFRAME (WINDOW_FRAME (w)))));
  1347.  
  1348.   set_buffer_internal (obuf);
  1349.   SET_PT (opoint);
  1350.   return Qnil;
  1351. }
  1352.  
  1353. DEFUN ("delete-windows-on", Fdelete_windows_on, Sdelete_windows_on,
  1354.   1, 1, "bDelete windows on (buffer): ",
  1355.   "Delete all windows showing BUFFER.\n\
  1356. Optional second argument FRAME controls which frames are affected.\n\
  1357. If nil or omitted, delete all windows showing BUFFER in any frame.\n\
  1358. If t, delete only windows showing BUFFER in the selected frame.\n\
  1359. If a frame, delete only windows showing BUFFER in that frame.")
  1360.   (buffer, frame)
  1361.      Lisp_Object buffer, frame;
  1362. {
  1363. #ifdef MULTI_FRAME
  1364.   /* FRAME uses t and nil to mean the opposite of what window_loop
  1365.      expects.  */
  1366.   if (! FRAMEP (frame))
  1367.     frame = NILP (frame) ? Qt : Qnil;
  1368. #else
  1369.   frame = Qt;
  1370. #endif
  1371.  
  1372.   if (!NILP (buffer))
  1373.     {
  1374.       buffer = Fget_buffer (buffer);
  1375.       CHECK_BUFFER (buffer, 0);
  1376.       window_loop (DELETE_BUFFER_WINDOWS, buffer, 0, frame);
  1377.     }
  1378.   return Qnil;
  1379. }
  1380.  
  1381. DEFUN ("replace-buffer-in-windows", Freplace_buffer_in_windows,
  1382.   Sreplace_buffer_in_windows,
  1383.   1, 1, "bReplace buffer in windows: ",
  1384.   "Replace BUFFER with some other buffer in all windows showing it.")
  1385.   (buffer)
  1386.      Lisp_Object buffer;
  1387. {
  1388.   if (!NILP (buffer))
  1389.     {
  1390.       buffer = Fget_buffer (buffer);
  1391.       CHECK_BUFFER (buffer, 0);
  1392.       window_loop (UNSHOW_BUFFER, buffer, 0, Qt);
  1393.     }
  1394.   return Qnil;
  1395. }
  1396.  
  1397. /* Set the height of WINDOW and all its inferiors.  */
  1398.  
  1399. /* The smallest acceptable dimensions for a window.  Anything smaller
  1400.    might crash Emacs.  */
  1401. #define MIN_SAFE_WINDOW_WIDTH  (2)
  1402. #define MIN_SAFE_WINDOW_HEIGHT (2)
  1403.  
  1404. /* Make sure that window_min_height and window_min_width are
  1405.    not too small; if they are, set them to safe minima.  */
  1406.  
  1407. static void
  1408. check_min_window_sizes ()
  1409. {
  1410.   /* Smaller values might permit a crash.  */
  1411.   if (window_min_width < MIN_SAFE_WINDOW_WIDTH)
  1412.     window_min_width = MIN_SAFE_WINDOW_WIDTH;
  1413.   if (window_min_height < MIN_SAFE_WINDOW_HEIGHT)
  1414.     window_min_height = MIN_SAFE_WINDOW_HEIGHT;
  1415. }
  1416.  
  1417. /* If *ROWS or *COLS are too small a size for FRAME, set them to the
  1418.    minimum allowable size.  */
  1419. void
  1420. check_frame_size (frame, rows, cols)
  1421.      FRAME_PTR frame;
  1422.      int *rows, *cols;
  1423. {
  1424.   /* For height, we have to see whether the frame has a minibuffer, and
  1425.      whether it wants a mode line.  */
  1426.   int min_height =
  1427.     (FRAME_MINIBUF_ONLY_P (frame) ? MIN_SAFE_WINDOW_HEIGHT - 1
  1428.      : (! FRAME_HAS_MINIBUF_P (frame)) ? MIN_SAFE_WINDOW_HEIGHT
  1429.      : 2 * MIN_SAFE_WINDOW_HEIGHT - 1);
  1430.  
  1431.   if (*rows < min_height)
  1432.     *rows = min_height;
  1433.   if (*cols  < MIN_SAFE_WINDOW_WIDTH)
  1434.     *cols = MIN_SAFE_WINDOW_WIDTH;
  1435. }
  1436.  
  1437. /* Normally the window is deleted if it gets too small.
  1438.    nodelete nonzero means do not do this.
  1439.    (The caller should check later and do so if appropriate)  */
  1440.  
  1441. _VOID_
  1442. set_window_height (window, height, nodelete)
  1443.      Lisp_Object window;
  1444.      int height;
  1445.      int nodelete;
  1446. {
  1447.   register struct window *w = XWINDOW (window);
  1448.   register struct window *c;
  1449.   int oheight = XFASTINT (w->height);
  1450.   int top, pos, lastbot, opos, lastobot;
  1451.   Lisp_Object child;
  1452.  
  1453.   check_min_window_sizes ();
  1454.  
  1455.   if (!nodelete
  1456.       && ! NILP (w->parent)
  1457.       && height < window_min_height)
  1458.     {
  1459.       Fdelete_window (window);
  1460.       return;
  1461.     }
  1462.  
  1463.   XFASTINT (w->last_modified) = 0;
  1464.   windows_or_buffers_changed++;
  1465.   XFASTINT (w->height) = height;
  1466.   if (!NILP (w->hchild))
  1467.     {
  1468.       for (child = w->hchild; !NILP (child); child = XWINDOW (child)->next)
  1469.     {
  1470.       XWINDOW (child)->top = w->top;
  1471.       set_window_height (child, height, nodelete);
  1472.     }
  1473.     }
  1474.   else if (!NILP (w->vchild))
  1475.     {
  1476.       lastbot = top = XFASTINT (w->top);
  1477.       lastobot = 0;
  1478.       for (child = w->vchild; !NILP (child); child = c->next)
  1479.     {
  1480.       c = XWINDOW (child);
  1481.  
  1482.       opos = lastobot + XFASTINT (c->height);
  1483.  
  1484.       XFASTINT (c->top) = lastbot;
  1485.  
  1486.       pos = (((opos * height) << 1) + oheight) / (oheight << 1);
  1487.  
  1488.       /* Avoid confusion: inhibit deletion of child if becomes too small */
  1489.       set_window_height (child, pos + top - lastbot, 1);
  1490.  
  1491.       /* Now advance child to next window,
  1492.          and set lastbot if child was not just deleted.  */
  1493.       lastbot = pos + top;
  1494.       lastobot = opos;
  1495.     }
  1496.       /* Now delete any children that became too small.  */
  1497.       if (!nodelete)
  1498.     for (child = w->vchild; !NILP (child); child = XWINDOW (child)->next)
  1499.       {
  1500.         set_window_height (child, XINT (XWINDOW (child)->height), 0);
  1501.       }
  1502.     }
  1503. }
  1504.  
  1505. /* Recursively set width of WINDOW and its inferiors. */
  1506.  
  1507. _VOID_
  1508. set_window_width (window, width, nodelete)
  1509.      Lisp_Object window;
  1510.      int width;
  1511.      int nodelete;
  1512. {
  1513.   register struct window *w = XWINDOW (window);
  1514.   register struct window *c;
  1515.   int owidth = XFASTINT (w->width);
  1516.   int left, pos, lastright, opos, lastoright;
  1517.   Lisp_Object child;
  1518.  
  1519.   if (!nodelete && width < window_min_width)
  1520.     {
  1521.       Fdelete_window (window);
  1522.       return;
  1523.     }
  1524.  
  1525.   XFASTINT (w->last_modified) = 0;
  1526.   windows_or_buffers_changed++;
  1527.   XFASTINT (w->width) = width;
  1528.   if (!NILP (w->vchild))
  1529.     {
  1530.       for (child = w->vchild; !NILP (child); child = XWINDOW (child)->next)
  1531.     {
  1532.       XWINDOW (child)->left = w->left;
  1533.       set_window_width (child, width, nodelete);
  1534.     }
  1535.     }
  1536.   else if (!NILP (w->hchild))
  1537.     {
  1538.       lastright = left = XFASTINT (w->left);
  1539.       lastoright = 0;
  1540.       for (child = w->hchild; !NILP (child); child = c->next)
  1541.     {
  1542.       c = XWINDOW (child);
  1543.  
  1544.       opos = lastoright + XFASTINT (c->width);
  1545.  
  1546.       XFASTINT (c->left) = lastright;
  1547.  
  1548.       pos = (((opos * width) << 1) + owidth) / (owidth << 1);
  1549.  
  1550.       /* Inhibit deletion for becoming too small */
  1551.       set_window_width (child, pos + left - lastright, 1);
  1552.  
  1553.       /* Now advance child to next window,
  1554.          and set lastright if child was not just deleted.  */
  1555.       lastright = pos + left, lastoright = opos;
  1556.     }
  1557.       /* Delete children that became too small */
  1558.       if (!nodelete)
  1559.     for (child = w->hchild; !NILP (child); child = XWINDOW (child)->next)
  1560.       {
  1561.         set_window_width (child, XINT (XWINDOW (child)->width), 0);
  1562.       }
  1563.     }
  1564. }
  1565.  
  1566. int window_select_count;
  1567.  
  1568. DEFUN ("set-window-buffer", Fset_window_buffer, Sset_window_buffer, 2, 2, 0,
  1569.   "Make WINDOW display BUFFER as its contents.\n\
  1570. BUFFER can be a buffer or buffer name.")
  1571.   (window, buffer)
  1572.      register Lisp_Object window, buffer;
  1573. {
  1574.   register Lisp_Object tem;
  1575.   register struct window *w = decode_window (window);
  1576.  
  1577.   buffer = Fget_buffer (buffer);
  1578.   CHECK_BUFFER (buffer, 1);
  1579.  
  1580.   if (NILP (XBUFFER (buffer)->name))
  1581.     error ("Attempt to display deleted buffer");
  1582.  
  1583.   tem = w->buffer;
  1584.   if (NILP (tem))
  1585.     error ("Window is deleted");
  1586.   else if (! EQ (tem, Qt))    /* w->buffer is t when the window
  1587.                    is first being set up.  */
  1588.     {
  1589.       if (!NILP (w->dedicated) && !EQ (tem, buffer))
  1590.     error ("Window is dedicated to %s\n", tem);
  1591.  
  1592.       unshow_buffer (w);
  1593.     }
  1594.  
  1595.   w->buffer = buffer;
  1596.   w->hscroll = 0;
  1597.   Fset_marker (w->pointm,
  1598.            make_number (BUF_PT (XBUFFER (buffer))),
  1599.            buffer);
  1600.   set_marker_restricted (w->start,
  1601.              make_number (XBUFFER (buffer)->last_window_start),
  1602.              buffer);
  1603.   w->start_at_line_beg = Qnil;
  1604.   w->force_start = Qnil;
  1605.   XFASTINT (w->last_modified) = 0;
  1606.   windows_or_buffers_changed++;
  1607.   if (EQ (window, selected_window))
  1608.     Fset_buffer (buffer);
  1609.  
  1610.   return Qnil;
  1611. }
  1612.  
  1613. DEFUN ("select-window", Fselect_window, Sselect_window, 1, 1, 0,
  1614.   "Select WINDOW.  Most editing will apply to WINDOW's buffer.\n\
  1615. The main editor command loop selects the buffer of the selected window\n\
  1616. before each command.")
  1617.   (window)
  1618.      register Lisp_Object window;
  1619. {
  1620.   register struct window *w;
  1621.   register struct window *ow = XWINDOW (selected_window);
  1622.  
  1623.   CHECK_LIVE_WINDOW (window, 0);
  1624.  
  1625.   w = XWINDOW (window);
  1626.  
  1627.   if (NILP (w->buffer))
  1628.     error ("Trying to select deleted window or non-leaf window");
  1629.  
  1630.   XFASTINT (w->use_time) = ++window_select_count;
  1631.   if (EQ (window, selected_window))
  1632.     return window;
  1633.  
  1634.   Fset_marker (ow->pointm, make_number (BUF_PT (XBUFFER (ow->buffer))),
  1635.            ow->buffer);
  1636.  
  1637.   selected_window = window;
  1638. #ifdef MULTI_FRAME
  1639.   if (XFRAME (WINDOW_FRAME (w)) != selected_frame)
  1640.     {
  1641.       XFRAME (WINDOW_FRAME (w))->selected_window = window;
  1642.       Fhandle_switch_frame (WINDOW_FRAME (w), Qnil);
  1643.     }
  1644.   else
  1645.     selected_frame->selected_window = window;
  1646. #endif
  1647.  
  1648.   record_buffer (w->buffer);
  1649.   Fset_buffer (w->buffer);
  1650.  
  1651.   /* Go to the point recorded in the window.
  1652.      This is important when the buffer is in more
  1653.      than one window.  It also matters when
  1654.      redisplay_window has altered point after scrolling,
  1655.      because it makes the change only in the window.  */
  1656.   {
  1657.     register int new_point = marker_position (w->pointm);
  1658.     if (new_point < BEGV)
  1659.       SET_PT (BEGV);
  1660.     if (new_point > ZV)
  1661.       SET_PT (ZV);
  1662.     else
  1663.       SET_PT (new_point);
  1664.   }
  1665.  
  1666.   windows_or_buffers_changed++;
  1667.   return window;
  1668. }
  1669.  
  1670. DEFUN ("display-buffer", Fdisplay_buffer, Sdisplay_buffer, 1, 2,
  1671.        "BDisplay buffer: \nP",
  1672.   "Make BUFFER appear in some window but don't select it.\n\
  1673. BUFFER can be a buffer or a buffer name.\n\
  1674. If BUFFER is shown already in some window, just use that one,\n\
  1675. unless the window is the selected window and the optional second\n\
  1676. argument NOT-THIS-WINDOW is non-nil (interactively, with prefix arg).\n\
  1677. Returns the window displaying BUFFER.")
  1678.   (buffer, not_this_window)
  1679.      register Lisp_Object buffer, not_this_window;
  1680. {
  1681.   register Lisp_Object window;
  1682.  
  1683.   buffer = Fget_buffer (buffer);
  1684.   CHECK_BUFFER (buffer, 0);
  1685.  
  1686.   if (!NILP (Vdisplay_buffer_function))
  1687.     return call2 (Vdisplay_buffer_function, buffer, not_this_window);
  1688.  
  1689.   if (NILP (not_this_window)
  1690.       && XBUFFER (XWINDOW (selected_window)->buffer) == XBUFFER (buffer))
  1691.     return selected_window;
  1692.  
  1693.   window = Fget_buffer_window (buffer, Qnil);
  1694.   if (!NILP (window)
  1695.       && (NILP (not_this_window) || !EQ (window, selected_window)))
  1696.     return window;
  1697.  
  1698. #ifdef MULTI_FRAME
  1699.   /* If there are no frames open that have more than a minibuffer,
  1700.      we need to create a new frame.  */
  1701.   if (pop_up_frames || last_nonminibuf_frame == 0)
  1702.     {
  1703.       window
  1704.     = Fframe_selected_window (call0 (Vpop_up_frame_function));
  1705.       Fset_window_buffer (window, buffer);
  1706. #if 0
  1707.       Fhandle_switch_frame (XWINDOW (window)->frame, Qnil);
  1708. #endif
  1709.       return window;
  1710.     }
  1711. #endif /* MULTI_FRAME */
  1712.  
  1713.   if (pop_up_windows
  1714. #ifdef MULTI_FRAME
  1715.       || FRAME_MINIBUF_ONLY_P (selected_frame)
  1716. #endif
  1717.       )
  1718.     {
  1719.       Lisp_Object frames = Qnil;
  1720.       
  1721. #ifdef MULTI_FRAME
  1722.       if (FRAME_MINIBUF_ONLY_P (selected_frame))
  1723.     XSET (frames, Lisp_Frame, last_nonminibuf_frame);
  1724. #endif
  1725.       /* Don't try to create a window if would get an error */
  1726.       if (split_height_threshold < window_min_height << 1)
  1727.     split_height_threshold = window_min_height << 1;
  1728.  
  1729.       window = Fget_largest_window (frames);
  1730.  
  1731.       if (!NILP (window)
  1732.       && window_height (window) >= split_height_threshold
  1733.       && (XFASTINT (XWINDOW (window)->width)
  1734.           == FRAME_WIDTH (XFRAME (WINDOW_FRAME (XWINDOW (window))))))
  1735.     window = Fsplit_window (window, Qnil, Qnil);
  1736.       else
  1737.     {
  1738.       window = Fget_lru_window (frames);
  1739.       if ((EQ (window, selected_window)
  1740.            || EQ (XWINDOW (window)->parent, Qnil))
  1741.           && window_height (window) >= window_min_height << 1)
  1742.         window = Fsplit_window (window, Qnil, Qnil);
  1743.     }
  1744.     }
  1745.   else
  1746.     window = Fget_lru_window (Qnil);
  1747.  
  1748.   Fset_window_buffer (window, buffer);
  1749.   return window;
  1750. }
  1751.  
  1752. void
  1753. temp_output_buffer_show (buf)
  1754.      register Lisp_Object buf;
  1755. {
  1756.   register struct buffer *old = current_buffer;
  1757.   register Lisp_Object window;
  1758.   register struct window *w;
  1759.  
  1760.   Fset_buffer (buf);
  1761.   XBUFFER (buf)->save_modified = MODIFF;
  1762.   BEGV = BEG;
  1763.   ZV = Z;
  1764.   SET_PT (BEG);
  1765.   clip_changed = 1;
  1766.   set_buffer_internal (old);
  1767.  
  1768.   if (!EQ (Vtemp_buffer_show_function, Qnil))
  1769.     call1 (Vtemp_buffer_show_function, buf);
  1770.   else
  1771.     {
  1772.       window = Fdisplay_buffer (buf, Qnil);
  1773.  
  1774. #ifdef MULTI_FRAME
  1775.       if (XFRAME (XWINDOW (window)->frame) != selected_frame)
  1776.     Fmake_frame_visible (WINDOW_FRAME (XWINDOW (window)));
  1777. #endif /* MULTI_FRAME */
  1778.       Vminibuf_scroll_window = window;
  1779.       w = XWINDOW (window);
  1780.       XFASTINT (w->hscroll) = 0;
  1781.       set_marker_restricted (w->start, make_number (1), buf);
  1782.       set_marker_restricted (w->pointm, make_number (1), buf);
  1783.     }
  1784. }
  1785.  
  1786. static _VOID_
  1787. make_dummy_parent (window)
  1788.      Lisp_Object window;
  1789. {
  1790.   register Lisp_Object old, new;
  1791.   register struct window *o, *p;
  1792.  
  1793.   old = window;
  1794.   XSETTYPE (old, Lisp_Vector);
  1795.   new = Fcopy_sequence (old);
  1796.   XSETTYPE (new, Lisp_Window);
  1797.  
  1798.   o = XWINDOW (old);
  1799.   p = XWINDOW (new);
  1800.   XFASTINT (p->sequence_number) = ++sequence_number;
  1801.  
  1802.   /* Put new into window structure in place of window */
  1803.   replace_window (window, new);
  1804.  
  1805.   o->next = Qnil;
  1806.   o->prev = Qnil;
  1807.   o->vchild = Qnil;
  1808.   o->hchild = Qnil;
  1809.   o->parent = new;
  1810.  
  1811.   p->start = Qnil;
  1812.   p->pointm = Qnil;
  1813.   p->buffer = Qnil;
  1814. }
  1815.  
  1816. DEFUN ("split-window", Fsplit_window, Ssplit_window, 0, 3, "",
  1817.   "Split WINDOW, putting SIZE lines in the first of the pair.\n\
  1818. WINDOW defaults to selected one and SIZE to half its size.\n\
  1819. If optional third arg HOR-FLAG is non-nil, split side by side\n\
  1820. and put SIZE columns in the first of the pair.")
  1821.   (window, chsize, horflag)
  1822.      Lisp_Object window, chsize, horflag;
  1823. {
  1824.   register Lisp_Object new;
  1825.   register struct window *o, *p;
  1826.   register int size;
  1827.  
  1828.   if (NILP (window))
  1829.     window = selected_window;
  1830.   else
  1831.     CHECK_LIVE_WINDOW (window, 0);
  1832.  
  1833.   o = XWINDOW (window);
  1834.  
  1835.   if (NILP (chsize))
  1836.     {
  1837.       if (!NILP (horflag))
  1838.     /* Round odd size up, since this is for the left-hand window,
  1839.        and it will lose a column for the separators.  */
  1840.     size = ((XFASTINT (o->width) + 1) & -2) >> 1;
  1841.       else
  1842.     size = XFASTINT (o->height) >> 1;
  1843.     }
  1844.   else
  1845.     {
  1846.       CHECK_NUMBER (chsize, 1);
  1847.       size = XINT (chsize);
  1848.     }
  1849.  
  1850.   if (MINI_WINDOW_P (o))
  1851.     error ("Attempt to split minibuffer window");
  1852.   else if (FRAME_NO_SPLIT_P (XFRAME (WINDOW_FRAME (o))))
  1853.     error ("Attempt to split unsplittable frame");
  1854.  
  1855.   check_min_window_sizes ();
  1856.  
  1857.   if (NILP (horflag))
  1858.     {
  1859.       if (size < window_min_height
  1860.       || size + window_min_height > XFASTINT (o->height))
  1861.     args_out_of_range_3 (window, chsize, horflag);
  1862.       if (NILP (o->parent)
  1863.       || NILP (XWINDOW (o->parent)->vchild))
  1864.     {
  1865.       make_dummy_parent (window);
  1866.       new = o->parent;
  1867.       XWINDOW (new)->vchild = window;
  1868.     }
  1869.     }
  1870.   else
  1871.     {
  1872.       if (size < window_min_width
  1873.       || size + window_min_width > XFASTINT (o->width))
  1874.     args_out_of_range_3 (window, chsize, horflag);
  1875.       if (NILP (o->parent)
  1876.       || NILP (XWINDOW (o->parent)->hchild))
  1877.     {
  1878.       make_dummy_parent (window);
  1879.       new = o->parent;
  1880.       XWINDOW (new)->hchild = window;
  1881.     }
  1882.     }
  1883.  
  1884.   /* Now we know that window's parent is a vertical combination
  1885.      if we are dividing vertically, or a horizontal combination
  1886.      if we are making side-by-side windows */
  1887.  
  1888.   windows_or_buffers_changed++;
  1889.   new = make_window ();
  1890.   p = XWINDOW (new);
  1891.  
  1892.   p->frame = o->frame;
  1893.   p->next = o->next;
  1894.   if (!NILP (p->next))
  1895.     XWINDOW (p->next)->prev = new;
  1896.   p->prev = window;
  1897.   o->next = new;
  1898.   p->parent = o->parent;
  1899.   p->buffer = Qt;
  1900.  
  1901.   Fset_window_buffer (new, o->buffer);
  1902.  
  1903.   /* Apportion the available frame space among the two new windows */
  1904.  
  1905.   if (!NILP (horflag))
  1906.     {
  1907.       p->height = o->height;
  1908.       p->top = o->top;
  1909.       XFASTINT (p->width) = XFASTINT (o->width) - size;
  1910.       XFASTINT (o->width) = size;
  1911.       XFASTINT (p->left) = XFASTINT (o->left) + size;
  1912.     }
  1913.   else
  1914.     {
  1915.       p->left = o->left;
  1916.       p->width = o->width;
  1917.       XFASTINT (p->height) = XFASTINT (o->height) - size;
  1918.       XFASTINT (o->height) = size;
  1919.       XFASTINT (p->top) = XFASTINT (o->top) + size;
  1920.     }
  1921.  
  1922.   return new;
  1923. }
  1924.  
  1925. DEFUN ("enlarge-window", Fenlarge_window, Senlarge_window, 1, 2, "p",
  1926.   "Make current window ARG lines bigger.\n\
  1927. From program, optional second arg non-nil means grow sideways ARG columns.")
  1928.   (n, side)
  1929.      register Lisp_Object n, side;
  1930. {
  1931.   CHECK_NUMBER (n, 0);
  1932.   change_window_height (XINT (n), !NILP (side));
  1933.   return Qnil;
  1934. }
  1935.  
  1936. DEFUN ("shrink-window", Fshrink_window, Sshrink_window, 1, 2, "p",
  1937.   "Make current window ARG lines smaller.\n\
  1938. From program, optional second arg non-nil means shrink sideways ARG columns.")
  1939.   (n, side)
  1940.      register Lisp_Object n, side;
  1941. {
  1942.   CHECK_NUMBER (n, 0);
  1943.   change_window_height (-XINT (n), !NILP (side));
  1944.   return Qnil;
  1945. }
  1946.  
  1947. int
  1948. window_height (window)
  1949.      Lisp_Object window;
  1950. {
  1951.   register struct window *p = XWINDOW (window);
  1952.   return XFASTINT (p->height);
  1953. }
  1954.  
  1955. int
  1956. window_width (window)
  1957.      Lisp_Object window;
  1958. {
  1959.   register struct window *p = XWINDOW (window);
  1960.   return XFASTINT (p->width);
  1961. }
  1962.  
  1963. #define MINSIZE(w)                        \
  1964.   (widthflag                            \
  1965.    ? window_min_width                        \
  1966.    : (MINI_WINDOW_P (XWINDOW (w)) ? 1 : window_min_height))
  1967.  
  1968. #define CURBEG(w) \
  1969.   *(widthflag ? (int *) &(XWINDOW (w)->left) : (int *) &(XWINDOW (w)->top))
  1970.  
  1971. #define CURSIZE(w) \
  1972.   *(widthflag ? (int *) &(XWINDOW (w)->width) : (int *) &(XWINDOW (w)->height))
  1973.  
  1974. /* Unlike set_window_height, this function
  1975.    also changes the heights of the siblings so as to
  1976.    keep everything consistent. */
  1977.  
  1978. _VOID_
  1979. change_window_height (delta, widthflag)
  1980.      register int delta;
  1981.      int widthflag;
  1982. {
  1983.   register Lisp_Object parent;
  1984.   Lisp_Object window;
  1985.   register struct window *p;
  1986.   int *sizep;
  1987.   int (*sizefun) () = widthflag ? window_width : window_height;
  1988.   register void (*setsizefun) _P_((Lisp_Object, int, int)) =
  1989.       (widthflag ? set_window_width : set_window_height);
  1990.  
  1991.   check_min_window_sizes ();
  1992.  
  1993.   window = selected_window;
  1994.   while (1)
  1995.     {
  1996.       p = XWINDOW (window);
  1997.       parent = p->parent;
  1998.       if (NILP (parent))
  1999.     {
  2000.       if (widthflag)
  2001.         error ("No other window to side of this one");
  2002.       break;
  2003.     }
  2004.       if (widthflag ? !NILP (XWINDOW (parent)->hchild)
  2005.       : !NILP (XWINDOW (parent)->vchild))
  2006.     break;
  2007.       window = parent;
  2008.     }
  2009.  
  2010.   sizep = &CURSIZE (window);
  2011.  
  2012.   if (*sizep + delta < MINSIZE (window))
  2013.     {
  2014.       Fdelete_window (window);
  2015.       return;
  2016.     }
  2017.  
  2018.   {
  2019.     register int maxdelta;
  2020.  
  2021.     maxdelta = (!NILP (parent) ? (*sizefun) (parent) - *sizep
  2022.         : !NILP (p->next) ? (*sizefun) (p->next) - MINSIZE (p->next)
  2023.         : !NILP (p->prev) ? (*sizefun) (p->prev) - MINSIZE (p->prev)
  2024.         /* This is a frame with only one window, a minibuffer-only
  2025.            or a minibufferless frame.  */
  2026.         : (delta = 0));
  2027.  
  2028.     if (delta > maxdelta)
  2029.       /* This case traps trying to make the minibuffer
  2030.      the full frame, or make the only window aside from the
  2031.      minibuffer the full frame.  */
  2032.       delta = maxdelta;
  2033.  
  2034.     if (delta == 0)
  2035.       return;
  2036.   }
  2037.  
  2038.   if (!NILP (p->next) &&
  2039.       (*sizefun) (p->next) - delta >= MINSIZE (p->next))
  2040.     {
  2041.       (*setsizefun) (p->next, (*sizefun) (p->next) - delta, 0);
  2042.       (*setsizefun) (window, *sizep + delta, 0);
  2043.       CURBEG (p->next) += delta;
  2044.       /* This does not change size of p->next,
  2045.      but it propagates the new top edge to its children */
  2046.       (*setsizefun) (p->next, (*sizefun) (p->next), 0);
  2047.     }
  2048.   else if (!NILP (p->prev) &&
  2049.        (*sizefun) (p->prev) - delta >= MINSIZE (p->prev))
  2050.     {
  2051.       (*setsizefun) (p->prev, (*sizefun) (p->prev) - delta, 0);
  2052.       CURBEG (window) -= delta;
  2053.       (*setsizefun) (window, *sizep + delta, 0);
  2054.     }
  2055.   else
  2056.     {
  2057.       register int delta1;
  2058.       register int opht = (*sizefun) (parent);
  2059.  
  2060.       /* If trying to grow this window to or beyond size of the parent,
  2061.      make delta1 so big that, on shrinking back down,
  2062.      all the siblings end up with less than one line and are deleted.  */
  2063.       if (opht <= *sizep + delta)
  2064.     delta1 = opht * opht * 2;
  2065.       /* Otherwise, make delta1 just right so that if we add delta1
  2066.      lines to this window and to the parent, and then shrink
  2067.      the parent back to its original size, the new proportional
  2068.      size of this window will increase by delta.  */
  2069.       else
  2070.     delta1 = (delta * opht * 100) / ((opht - *sizep - delta) * 100);
  2071.  
  2072.       /* Add delta1 lines or columns to this window, and to the parent,
  2073.      keeping things consistent while not affecting siblings.  */
  2074.       CURSIZE (parent) = opht + delta1;
  2075.       (*setsizefun) (window, *sizep + delta1, 0);
  2076.  
  2077.       /* Squeeze out delta1 lines or columns from our parent,
  2078.      shriking this window and siblings proportionately.
  2079.      This brings parent back to correct size.
  2080.      Delta1 was calculated so this makes this window the desired size,
  2081.      taking it all out of the siblings.  */
  2082.       (*setsizefun) (parent, opht, 0);
  2083.     }
  2084.  
  2085.   XFASTINT (p->last_modified) = 0;
  2086. }
  2087. #undef MINSIZE
  2088. #undef CURBEG
  2089. #undef CURSIZE
  2090.  
  2091.  
  2092. /* Return number of lines of text (not counting mode line) in W.  */
  2093.  
  2094. int
  2095. window_internal_height (w)
  2096.      struct window *w;
  2097. {
  2098.   int ht = XFASTINT (w->height);
  2099.  
  2100.   if (MINI_WINDOW_P (w))
  2101.     return ht;
  2102.  
  2103.   if (!NILP (w->parent) || !NILP (w->vchild) || !NILP (w->hchild)
  2104.       || !NILP (w->next) || !NILP (w->prev)
  2105.       || FRAME_WANTS_MODELINE_P (XFRAME (WINDOW_FRAME (w))))
  2106.     return ht - 1;
  2107.  
  2108.   return ht;
  2109. }
  2110.  
  2111.  
  2112. /* Return the number of columns in W.
  2113.    Don't count columns occupied by scroll bars or the vertical bar
  2114.    separating W from the sibling to its right.  */
  2115. int
  2116. window_internal_width (w)
  2117.      struct window *w;
  2118. {
  2119.   FRAME_PTR f = XFRAME (WINDOW_FRAME (w));
  2120.   int left = XINT (w->left);
  2121.   int width = XINT (w->width);
  2122.  
  2123.   /* If this window is flush against the right edge of the frame, its
  2124.      internal width is its full width.  */
  2125.   if (left + width >= FRAME_WIDTH (f))
  2126.     return width;
  2127.  
  2128.   /* If we are not flush right, then our rightmost columns are
  2129.      occupied by some sort of separator.  */
  2130.  
  2131.   /* Scroll bars occupy a few columns.  */
  2132.   if (FRAME_HAS_VERTICAL_SCROLL_BARS (f))
  2133.     return width - VERTICAL_SCROLL_BAR_WIDTH;
  2134.  
  2135.   /* The column of `|' characters separating side-by-side windows
  2136.      occupies one column only.  */
  2137.   return width - 1;
  2138. }
  2139.  
  2140.  
  2141. /* Scroll contents of window WINDOW up N lines.  */
  2142.  
  2143. void
  2144. window_scroll (window, n, noerror)
  2145.      Lisp_Object window;
  2146.      int n;
  2147.      int noerror;
  2148. {
  2149.   register struct window *w = XWINDOW (window);
  2150.   register int opoint = point;
  2151.   register int pos;
  2152.   register int ht = window_internal_height (w);
  2153.   register Lisp_Object tem;
  2154.   int lose;
  2155.   Lisp_Object bolp;
  2156.  
  2157.   XFASTINT (tem) = point;
  2158.   tem = Fpos_visible_in_window_p (tem, window);
  2159.  
  2160.   if (NILP (tem))
  2161.     {
  2162.       Fvertical_motion (make_number (- ht / 2));
  2163.       XFASTINT (tem) = point;
  2164.       Fset_marker (w->start, tem, w->buffer);
  2165.       w->force_start = Qt;
  2166.     }
  2167.  
  2168.   SET_PT (marker_position (w->start));
  2169.   lose = n < 0 && point == BEGV;
  2170.   Fvertical_motion (make_number (n));
  2171.   pos = point;
  2172.   bolp = Fbolp ();
  2173.   SET_PT (opoint);
  2174.  
  2175.   if (lose)
  2176.     {
  2177.       if (noerror)
  2178.     return;
  2179.       else
  2180.     Fsignal (Qbeginning_of_buffer, Qnil);
  2181.     }
  2182.  
  2183.   if (pos < ZV)
  2184.     {
  2185.       set_marker_restricted (w->start, make_number (pos), w->buffer);
  2186.       w->start_at_line_beg = bolp;
  2187.       w->update_mode_line = Qt;
  2188.       XFASTINT (w->last_modified) = 0;
  2189.       if (pos > opoint)
  2190.     SET_PT (pos);
  2191.       if (n < 0)
  2192.     {
  2193.       SET_PT (pos);
  2194.       tem = Fvertical_motion (make_number (ht));
  2195.       if (point > opoint || XFASTINT (tem) < ht)
  2196.         SET_PT (opoint);
  2197.       else
  2198.         Fvertical_motion (make_number (-1));
  2199.     }
  2200.     }
  2201.   else
  2202.     {
  2203.       if (noerror)
  2204.     return;
  2205.       else
  2206.     Fsignal (Qend_of_buffer, Qnil);
  2207.     }
  2208. }
  2209.  
  2210. /* This is the guts of Fscroll_up and Fscroll_down.  */
  2211.  
  2212. static void
  2213. scroll_command (n, direction)
  2214.      register Lisp_Object n;
  2215.      int direction;
  2216. {
  2217.   register int defalt;
  2218.   int count = specpdl_ptr - specpdl;
  2219.  
  2220.   /* If selected window's buffer isn't current, make it current for the moment.
  2221.      But don't screw up if window_scroll gets an error.  */
  2222.   if (XBUFFER (XWINDOW (selected_window)->buffer) != current_buffer)
  2223.     {
  2224.       record_unwind_protect (save_excursion_restore, save_excursion_save ());
  2225.       Fset_buffer (XWINDOW (selected_window)->buffer);
  2226.     }
  2227.  
  2228.   defalt = (window_internal_height (XWINDOW (selected_window))
  2229.         - next_screen_context_lines);
  2230.   defalt = direction * (defalt < 1 ? 1 : defalt);
  2231.  
  2232.   if (NILP (n))
  2233.     window_scroll (selected_window, defalt, 0);
  2234.   else if (EQ (n, Qminus))
  2235.     window_scroll (selected_window, - defalt, 0);
  2236.   else
  2237.     {
  2238.       n = Fprefix_numeric_value (n);
  2239.       window_scroll (selected_window, XINT (n) * direction, 0);
  2240.     }
  2241.  
  2242.   unbind_to (count, Qnil);
  2243. }
  2244.  
  2245. DEFUN ("scroll-up", Fscroll_up, Sscroll_up, 0, 1, "P",
  2246.   "Scroll text of current window upward ARG lines; or near full screen if no ARG.\n\
  2247. A near full screen is `next-screen-context-lines' less than a full screen.\n\
  2248. When calling from a program, supply a number as argument or nil.")
  2249.   (n)
  2250.      Lisp_Object n;
  2251. {
  2252.   scroll_command (n, 1);
  2253.   return Qnil;
  2254. }
  2255.  
  2256. DEFUN ("scroll-down", Fscroll_down, Sscroll_down, 0, 1, "P",
  2257.   "Scroll text of current window downward ARG lines; or near full screen if no ARG.\n\
  2258. A near full screen is `next-screen-context-lines' less than a full screen.\n\
  2259. When calling from a program, supply a number as argument or nil.")
  2260.   (n)
  2261.      Lisp_Object n;
  2262. {
  2263.   scroll_command (n, -1);
  2264.   return Qnil;
  2265. }
  2266.  
  2267. DEFUN ("scroll-other-window", Fscroll_other_window, Sscroll_other_window, 0, 1, "P",
  2268.   "Scroll next window upward ARG lines; or near full screen if no ARG.\n\
  2269. The next window is the one below the current one; or the one at the top\n\
  2270. if the current one is at the bottom.\n\
  2271. When calling from a program, supply a number as argument or nil.\n\
  2272. \n\
  2273. If in the minibuffer, `minibuf-scroll-window' if non-nil\n\
  2274. specifies the window to scroll.\n\
  2275. If `other-window-scroll-buffer' is non-nil, scroll the window\n\
  2276. showing that buffer, popping the buffer up if necessary.")
  2277.   (n)
  2278.      register Lisp_Object n;
  2279. {
  2280.   register Lisp_Object window;
  2281.   register int ht;
  2282.   register struct window *w;
  2283.   register int count = specpdl_ptr - specpdl;
  2284.  
  2285.   if (MINI_WINDOW_P (XWINDOW (selected_window))
  2286.       && !NILP (Vminibuf_scroll_window))
  2287.     window = Vminibuf_scroll_window;
  2288.   /* If buffer is specified, scroll that buffer.  */
  2289.   else if (!NILP (Vother_window_scroll_buffer))
  2290.     {
  2291.       window = Fget_buffer_window (Vother_window_scroll_buffer, Qnil);
  2292.       if (NILP (window))
  2293.     window = Fdisplay_buffer (Vother_window_scroll_buffer, Qt);
  2294.     }
  2295.   else
  2296.     {
  2297.       /* Nothing specified; look for a neighboring window on the same
  2298.      frame.  */
  2299.       window = Fnext_window (selected_window, Qnil, Qnil);
  2300.  
  2301.       if (EQ (window, selected_window))
  2302.     /* That didn't get us anywhere; look for a window on another
  2303.            visible frame.  */
  2304.     do
  2305.       window = Fnext_window (window, Qnil, Qt);
  2306.     while (! FRAME_VISIBLE_P (XFRAME (WINDOW_FRAME (XWINDOW (window))))
  2307.            && ! EQ (window, selected_window));
  2308.     }
  2309.  
  2310.   CHECK_LIVE_WINDOW (window, 0);
  2311.  
  2312.   if (EQ (window, selected_window))
  2313.     error ("There is no other window");
  2314.  
  2315.   w = XWINDOW (window);
  2316.   ht = window_internal_height (w);
  2317.  
  2318.   /* Don't screw up if window_scroll gets an error.  */
  2319.   record_unwind_protect (save_excursion_restore, save_excursion_save ());
  2320.  
  2321.   Fset_buffer (w->buffer);
  2322.   SET_PT (marker_position (w->pointm));
  2323.  
  2324.   if (NILP (n))
  2325.     window_scroll (window, ht - next_screen_context_lines, 1);
  2326.   else if (EQ (n, Qminus))
  2327.     window_scroll (window, next_screen_context_lines - ht, 1);
  2328.   else
  2329.     {
  2330.       if (XTYPE (n) == Lisp_Cons)
  2331.     n = Fcar (n);
  2332.       CHECK_NUMBER (n, 0);
  2333.       window_scroll (window, XINT (n), 1);
  2334.     }
  2335.  
  2336.   Fset_marker (w->pointm, make_number (point), Qnil);
  2337.   unbind_to (count, Qnil);
  2338.  
  2339.   return Qnil;
  2340. }
  2341.  
  2342. DEFUN ("scroll-left", Fscroll_left, Sscroll_left, 0, 1, "P",
  2343.   "Scroll selected window display ARG columns left.\n\
  2344. Default for ARG is window width minus 2.")
  2345.   (arg)
  2346.      register Lisp_Object arg;
  2347. {
  2348.  
  2349.   if (NILP (arg))
  2350.     XFASTINT (arg) = window_internal_width (XWINDOW (selected_window)) - 2;
  2351.   else
  2352.     arg = Fprefix_numeric_value (arg);
  2353.  
  2354.   return
  2355.     Fset_window_hscroll (selected_window,
  2356.              make_number (XINT (XWINDOW (selected_window)->hscroll)
  2357.                       + XINT (arg)));
  2358. }
  2359.  
  2360. DEFUN ("scroll-right", Fscroll_right, Sscroll_right, 0, 1, "P",
  2361.   "Scroll selected window display ARG columns right.\n\
  2362. Default for ARG is window width minus 2.")
  2363.   (arg)
  2364.      register Lisp_Object arg;
  2365. {
  2366.   if (NILP (arg))
  2367.     XFASTINT (arg) = window_internal_width (XWINDOW (selected_window)) - 2;
  2368.   else
  2369.     arg = Fprefix_numeric_value (arg);
  2370.  
  2371.   return
  2372.     Fset_window_hscroll (selected_window,
  2373.              make_number (XINT (XWINDOW (selected_window)->hscroll)
  2374.                       - XINT (arg)));
  2375. }
  2376.  
  2377. DEFUN ("recenter", Frecenter, Srecenter, 0, 1, "P",
  2378.   "Center point in window and redisplay frame.  With ARG, put point on line ARG.\n\
  2379. The desired position of point is always relative to the current window.\n\
  2380. Just C-u as prefix means put point in the center of the window.\n\
  2381. No arg (i.e., it is nil) erases the entire frame and then\n\
  2382. redraws with point in the center of the current window.")
  2383.   (n)
  2384.      register Lisp_Object n;
  2385. {
  2386.   register struct window *w = XWINDOW (selected_window);
  2387.   register int ht = window_internal_height (w);
  2388.   register int opoint = point;
  2389.  
  2390.   if (NILP (n))
  2391.     {
  2392.       extern int frame_garbaged;
  2393.  
  2394.       SET_FRAME_GARBAGED (XFRAME (WINDOW_FRAME (w)));
  2395.       XFASTINT (n) = ht / 2;
  2396.     }
  2397.   else if (XTYPE (n) == Lisp_Cons) /* Just C-u. */
  2398.     {
  2399.       XFASTINT (n) = ht / 2;
  2400.     }
  2401.   else
  2402.     {
  2403.       n = Fprefix_numeric_value (n);
  2404.       CHECK_NUMBER (n, 0);
  2405.     }
  2406.  
  2407.   if (XINT (n) < 0)
  2408.     XSETINT (n, XINT (n) + ht);
  2409.  
  2410.   XSETINT (n, - XINT (n));
  2411.  
  2412.   Fvertical_motion (n);
  2413.   Fset_marker (w->start, make_number (point), w->buffer);
  2414.   w->start_at_line_beg = Fbolp ();
  2415.  
  2416.   SET_PT (opoint);
  2417.   w->force_start = Qt;
  2418.  
  2419.   return Qnil;
  2420. }
  2421.  
  2422. DEFUN ("move-to-window-line", Fmove_to_window_line, Smove_to_window_line,
  2423.   1, 1, "P",
  2424.   "Position point relative to window.\n\
  2425. With no argument, position text at center of window.\n\
  2426. An argument specifies frame line; zero means top of window,\n\
  2427. negative means relative to bottom of window.")
  2428.   (arg)
  2429.      register Lisp_Object arg;
  2430. {
  2431.   register struct window *w = XWINDOW (selected_window);
  2432.   register int height = window_internal_height (w);
  2433.   register int start;
  2434.  
  2435.   if (NILP (arg))
  2436.     XFASTINT (arg) = height / 2;
  2437.   else
  2438.     {
  2439.       arg = Fprefix_numeric_value (arg);
  2440.       if (XINT (arg) < 0)
  2441.     XSETINT (arg, XINT (arg) + height);
  2442.     }
  2443.  
  2444.   start = marker_position (w->start);
  2445.   if (start < BEGV || start > ZV)
  2446.     {
  2447.       Fvertical_motion (make_number (- height / 2));
  2448.       Fset_marker (w->start, make_number (point), w->buffer);
  2449.       w->start_at_line_beg = Fbolp ();
  2450.       w->force_start = Qt;
  2451.     }
  2452.   else
  2453.     SET_PT (start);
  2454.  
  2455.   return Fvertical_motion (arg);
  2456. }
  2457.  
  2458. struct save_window_data
  2459.   {
  2460.     int size_from_Lisp_Vector_struct;
  2461.     struct Lisp_Vector *next_from_Lisp_Vector_struct;
  2462.     Lisp_Object frame_width, frame_height;
  2463.     Lisp_Object selected_frame;
  2464.     Lisp_Object current_window;
  2465.     Lisp_Object current_buffer;
  2466.     Lisp_Object minibuf_scroll_window;
  2467.     Lisp_Object root_window;
  2468.     Lisp_Object focus_frame;
  2469.     /* A vector, interpreted as a struct saved_window */
  2470.     Lisp_Object saved_windows;
  2471.   };
  2472.  
  2473. /* Arg to Fmake_vector */
  2474. #define SAVE_WINDOW_DATA_SIZE                        \
  2475.   ((sizeof (struct save_window_data)                    \
  2476.     - (sizeof (struct Lisp_Vector)                    \
  2477.        /* Don't count the contents member of the struct Lisp_Vector */    \
  2478.        - sizeof (Lisp_Object)))                        \
  2479.    / sizeof (Lisp_Object))
  2480.  
  2481. /* This is saved as a Lisp_Vector */
  2482. struct saved_window
  2483.   {
  2484.     /* these first two must agree with struct Lisp_Vector in lisp.h */
  2485.     int size_from_Lisp_Vector_struct;
  2486.     struct Lisp_Vector *next_from_Lisp_Vector_struct;
  2487.  
  2488.     Lisp_Object window;
  2489.     Lisp_Object buffer, start, pointm, mark;
  2490.     Lisp_Object left, top, width, height, hscroll;
  2491.     Lisp_Object parent, prev;
  2492.     Lisp_Object start_at_line_beg;
  2493.     Lisp_Object display_table;
  2494.   };
  2495. #define SAVED_WINDOW_VECTOR_SIZE 14 /* Arg to Fmake_vector */
  2496.  
  2497. #define SAVED_WINDOW_N(swv,n) \
  2498.   ((struct saved_window *) (XVECTOR ((swv)->contents[(n)])))
  2499.  
  2500. DEFUN ("window-configuration-p", Fwindow_configuration_p, Swindow_configuration_p, 1, 1, 0,
  2501.   "T if OBJECT is a window-configration object.")
  2502.   (obj)
  2503.      Lisp_Object obj;
  2504. {
  2505.   if (XTYPE (obj) == Lisp_Window_Configuration)
  2506.     return Qt;
  2507.   return Qnil;
  2508. }
  2509.  
  2510.  
  2511. DEFUN ("set-window-configuration", Fset_window_configuration,
  2512.   Sset_window_configuration, 1, 1, 0,
  2513.   "Set the configuration of windows and buffers as specified by CONFIGURATION.\n\
  2514. CONFIGURATION must be a value previously returned\n\
  2515. by `current-window-configuration' (which see).")
  2516.      (configuration)
  2517.      Lisp_Object configuration;
  2518. {
  2519.   register struct save_window_data *data;
  2520.   struct Lisp_Vector *saved_windows;
  2521.   Lisp_Object new_current_buffer;
  2522.   Lisp_Object frame;
  2523.   FRAME_PTR f;
  2524.  
  2525.   while (XTYPE (configuration) != Lisp_Window_Configuration)
  2526.     {
  2527.       configuration = wrong_type_argument (intern ("window-configuration-p"),
  2528.                        configuration);
  2529.     }
  2530.  
  2531.   data = (struct save_window_data *) XVECTOR (configuration);
  2532.   saved_windows = XVECTOR (data->saved_windows);
  2533.  
  2534.   new_current_buffer = data->current_buffer;
  2535.   if (NILP (XBUFFER (new_current_buffer)->name))
  2536.     new_current_buffer = Qnil;
  2537.  
  2538.   frame = XWINDOW (SAVED_WINDOW_N (saved_windows, 0)->window)->frame;
  2539.   f = XFRAME (frame);
  2540.  
  2541.   /* If f is a dead frame, don't bother rebuilding its window tree.
  2542.      However, there is other stuff we should still try to do below.  */
  2543.   if (FRAME_LIVE_P (f))
  2544.     {
  2545.       register struct window *w;
  2546.       register struct saved_window *p;
  2547.       int k;
  2548.  
  2549.       /* If the frame has been resized since this window configuration was
  2550.      made, we change the frame to the size specified in the
  2551.      configuration, restore the configuration, and then resize it
  2552.      back.  We keep track of the prevailing height in these variables.  */
  2553.       int previous_frame_height = FRAME_HEIGHT (f);
  2554.       int previous_frame_width =  FRAME_WIDTH  (f);
  2555.  
  2556.       if (XFASTINT (data->frame_height) != previous_frame_height
  2557.       || XFASTINT (data->frame_width) != previous_frame_width)
  2558.     change_frame_size (f, data->frame_height, data->frame_width, 0, 0);
  2559.  
  2560.       windows_or_buffers_changed++;
  2561.  
  2562.       /* Kludge Alert!
  2563.      Mark all windows now on frame as "deleted".
  2564.      Restoring the new configuration "undeletes" any that are in it.
  2565.      
  2566.      Save their current buffers in their height fields, since we may
  2567.      need it later, if a buffer saved in the configuration is now
  2568.      dead.  */
  2569.       delete_all_subwindows (XWINDOW (FRAME_ROOT_WINDOW (f)));
  2570.  
  2571.       for (k = 0; k < saved_windows->size; k++)
  2572.     {
  2573.       p = SAVED_WINDOW_N (saved_windows, k);
  2574.       w = XWINDOW (p->window);
  2575.       w->next = Qnil;
  2576.  
  2577.       if (!NILP (p->parent))
  2578.         w->parent = SAVED_WINDOW_N (saved_windows,
  2579.                     XFASTINT (p->parent))->window;
  2580.       else
  2581.         w->parent = Qnil;
  2582.  
  2583.       if (!NILP (p->prev))
  2584.         {
  2585.           w->prev = SAVED_WINDOW_N (saved_windows,
  2586.                     XFASTINT (p->prev))->window;
  2587.           XWINDOW (w->prev)->next = p->window;
  2588.         }
  2589.       else
  2590.         {
  2591.           w->prev = Qnil;
  2592.           if (!NILP (w->parent))
  2593.         {
  2594.           if (EQ (p->width, XWINDOW (w->parent)->width))
  2595.             {
  2596.               XWINDOW (w->parent)->vchild = p->window;
  2597.               XWINDOW (w->parent)->hchild = Qnil;
  2598.             }
  2599.           else
  2600.             {
  2601.               XWINDOW (w->parent)->hchild = p->window;
  2602.               XWINDOW (w->parent)->vchild = Qnil;
  2603.             }
  2604.         }
  2605.         }
  2606.  
  2607.       /* If we squirreled away the buffer in the window's height,
  2608.          restore it now.  */
  2609.       if (XTYPE (w->height) == Lisp_Buffer)
  2610.         w->buffer = w->height;
  2611.       w->left = p->left;
  2612.       w->top = p->top;
  2613.       w->width = p->width;
  2614.       w->height = p->height;
  2615.       w->hscroll = p->hscroll;
  2616.       w->display_table = p->display_table;
  2617.       XFASTINT (w->last_modified) = 0;
  2618.  
  2619.       /* Reinstall the saved buffer and pointers into it.  */
  2620.       if (NILP (p->buffer))
  2621.         w->buffer = p->buffer;
  2622.       else
  2623.         {
  2624.           if (!NILP (XBUFFER (p->buffer)->name))
  2625.         /* If saved buffer is alive, install it.  */
  2626.         {
  2627.           w->buffer = p->buffer;
  2628.           w->start_at_line_beg = p->start_at_line_beg;
  2629.           set_marker_restricted (w->start,
  2630.                      Fmarker_position (p->start),
  2631.                      w->buffer);
  2632.           set_marker_restricted (w->pointm,
  2633.                      Fmarker_position (p->pointm),
  2634.                      w->buffer);
  2635.           Fset_marker (XBUFFER (w->buffer)->mark,
  2636.                    Fmarker_position (p->mark), w->buffer);
  2637.  
  2638.           /* As documented in Fcurrent_window_configuration, don't
  2639.              save the location of point in the buffer which was current
  2640.              when the window configuration was recorded.  */
  2641.           if (!EQ (p->buffer, new_current_buffer) &&
  2642.               XBUFFER (p->buffer) == current_buffer)
  2643.             Fgoto_char (w->pointm);
  2644.         }
  2645.           else if (NILP (w->buffer) || NILP (XBUFFER (w->buffer)->name))
  2646.         /* Else unless window has a live buffer, get one.  */
  2647.         {
  2648.           w->buffer = Fcdr (Fcar (Vbuffer_alist));
  2649.           /* This will set the markers to beginning of visible
  2650.              range.  */
  2651.           set_marker_restricted (w->start, make_number (0), w->buffer);
  2652.           set_marker_restricted (w->pointm, make_number (0),w->buffer);
  2653.           w->start_at_line_beg = Qt;
  2654.         }
  2655.           else
  2656.         /* Keeping window's old buffer; make sure the markers
  2657.            are real.  */
  2658.         {
  2659.           /* Set window markers at start of visible range.  */
  2660.           if (XMARKER (w->start)->buffer == 0)
  2661.             set_marker_restricted (w->start, make_number (0),
  2662.                        w->buffer);
  2663.           if (XMARKER (w->pointm)->buffer == 0)
  2664.             set_marker_restricted (w->pointm,
  2665.                        (make_number
  2666.                         (BUF_PT (XBUFFER (w->buffer)))),
  2667.                        w->buffer);
  2668.           w->start_at_line_beg = Qt;
  2669.         }
  2670.         }
  2671.     }
  2672.  
  2673.       FRAME_ROOT_WINDOW (f) = data->root_window;
  2674.       Fselect_window (data->current_window);
  2675.  
  2676. #ifdef MULTI_FRAME
  2677.       if (NILP (data->focus_frame)
  2678.       || (XTYPE (data->focus_frame) == Lisp_Frame
  2679.           && FRAME_LIVE_P (XFRAME (data->focus_frame))))
  2680.     Fredirect_frame_focus (frame, data->focus_frame);
  2681. #endif
  2682.  
  2683. #if 0 /* I don't understand why this is needed, and it causes problems
  2684.          when the frame's old selected window has been deleted.  */
  2685. #ifdef MULTI_FRAME
  2686.       if (f != selected_frame && ! FRAME_TERMCAP_P (f))
  2687.     Fhandle_switch_frame (WINDOW_FRAME (XWINDOW (data->root_window)), Qnil);
  2688. #endif
  2689. #endif
  2690.  
  2691.       /* Set the screen height to the value it had before this function.  */
  2692.       if (previous_frame_height != FRAME_HEIGHT (f)
  2693.       || previous_frame_width != FRAME_WIDTH (f))
  2694.     change_frame_size (f, previous_frame_height, previous_frame_width,
  2695.                0, 0);
  2696.     }
  2697.  
  2698. #ifdef MULTI_FRAME
  2699.   /* Fselect_window will have made f the selected frame, so we
  2700.      reselect the proper frame here.  Fhandle_switch_frame will change the
  2701.      selected window too, but that doesn't make the call to
  2702.      Fselect_window above totally superfluous; it still sets f's
  2703.      selected window.  */
  2704.   if (FRAME_LIVE_P (XFRAME (data->selected_frame)))
  2705.     Fhandle_switch_frame (data->selected_frame, Qnil);
  2706. #endif
  2707.  
  2708.   if (!NILP (new_current_buffer))
  2709.     Fset_buffer (new_current_buffer);
  2710.  
  2711.   Vminibuf_scroll_window = data->minibuf_scroll_window;
  2712.   return (Qnil);
  2713. }
  2714.  
  2715. /* Mark all windows now on frame as deleted
  2716.    by setting their buffers to nil.  */
  2717.  
  2718. void
  2719. delete_all_subwindows (w)
  2720.      register struct window *w;
  2721. {
  2722.   if (!NILP (w->next))
  2723.     delete_all_subwindows (XWINDOW (w->next));
  2724.   if (!NILP (w->vchild))
  2725.     delete_all_subwindows (XWINDOW (w->vchild));
  2726.   if (!NILP (w->hchild))
  2727.     delete_all_subwindows (XWINDOW (w->hchild));
  2728.  
  2729.   w->height = w->buffer;       /* See Fset_window_configuration for excuse.  */
  2730.  
  2731.   /* We set all three of these fields to nil, to make sure that we can
  2732.      distinguish this dead window from any live window.  Live leaf
  2733.      windows will have buffer set, and combination windows will have
  2734.      vchild or hchild set.  */
  2735.   w->buffer = Qnil;
  2736.   w->vchild = Qnil;
  2737.   w->hchild = Qnil;
  2738. }
  2739.  
  2740. static int
  2741. count_windows (window)
  2742.      register struct window *window;
  2743. {
  2744.   register int count = 1;
  2745.   if (!NILP (window->next))
  2746.     count += count_windows (XWINDOW (window->next));
  2747.   if (!NILP (window->vchild))
  2748.     count += count_windows (XWINDOW (window->vchild));
  2749.   if (!NILP (window->hchild))
  2750.     count += count_windows (XWINDOW (window->hchild));
  2751.   return count;
  2752. }
  2753.  
  2754. static int
  2755. save_window_save (window, vector, i)
  2756.      Lisp_Object window;
  2757.      struct Lisp_Vector *vector;
  2758.      int i;
  2759. {
  2760.   register struct saved_window *p;
  2761.   register struct window *w;
  2762.   register Lisp_Object tem;
  2763.  
  2764.   for (;!NILP (window); window = w->next)
  2765.     {
  2766.       p = SAVED_WINDOW_N (vector, i);
  2767.       w = XWINDOW (window);
  2768.  
  2769.       XFASTINT (w->temslot) = i++;
  2770.       p->window = window;
  2771.       p->buffer = w->buffer;
  2772.       p->left = w->left;
  2773.       p->top = w->top;
  2774.       p->width = w->width;
  2775.       p->height = w->height;
  2776.       p->hscroll = w->hscroll;
  2777.       p->display_table = w->display_table;
  2778.       if (!NILP (w->buffer))
  2779.     {
  2780.       /* Save w's value of point in the window configuration.
  2781.          If w is the selected window, then get the value of point
  2782.          from the buffer; pointm is garbage in the selected window.  */
  2783.       if (EQ (window, selected_window))
  2784.         {
  2785.           p->pointm = Fmake_marker ();
  2786.           Fset_marker (p->pointm, BUF_PT (XBUFFER (w->buffer)),
  2787.                w->buffer);
  2788.         }
  2789.       else
  2790.         p->pointm = Fcopy_marker (w->pointm);
  2791.  
  2792.       p->start = Fcopy_marker (w->start);
  2793.       p->start_at_line_beg = w->start_at_line_beg;
  2794.  
  2795.       tem = XBUFFER (w->buffer)->mark;
  2796.       p->mark = Fcopy_marker (tem);
  2797.     }
  2798.       else
  2799.     {
  2800.       p->pointm = Qnil;
  2801.       p->start = Qnil;
  2802.       p->mark = Qnil;
  2803.       p->start_at_line_beg = Qnil;
  2804.     }
  2805.  
  2806.       if (NILP (w->parent))
  2807.     p->parent = Qnil;
  2808.       else
  2809.     p->parent = XWINDOW (w->parent)->temslot;
  2810.  
  2811.       if (NILP (w->prev))
  2812.     p->prev = Qnil;
  2813.       else
  2814.     p->prev = XWINDOW (w->prev)->temslot;
  2815.  
  2816.       if (!NILP (w->vchild))
  2817.     i = save_window_save (w->vchild, vector, i);
  2818.       if (!NILP (w->hchild))
  2819.     i = save_window_save (w->hchild, vector, i);
  2820.     }
  2821.  
  2822.   return i;
  2823. }
  2824.  
  2825. DEFUN ("current-window-configuration",
  2826.     Fcurrent_window_configuration, Scurrent_window_configuration, 0, 1, 0,
  2827.   "Return an object representing the current window configuration of FRAME.\n\
  2828. If FRAME is nil or omitted, use the selected frame.\n\
  2829. This describes the number of windows, their sizes and current buffers,\n\
  2830. and for each displayed buffer, where display starts, and the positions of\n\
  2831. point and mark.  An exception is made for point in the current buffer:\n\
  2832. its value is -not- saved.\n\
  2833. This also records the currently selected frame, and FRAME's focus\n\
  2834. redirection (see `redirect-frame-focus').")
  2835.   (frame)
  2836.      Lisp_Object frame;
  2837. {
  2838.   register Lisp_Object tem;
  2839.   register int n_windows;
  2840.   register struct save_window_data *data;
  2841.   register int i;
  2842.   FRAME_PTR f;
  2843.  
  2844.   if (NILP (frame))
  2845.     f = selected_frame;
  2846.   else
  2847.     {
  2848.       CHECK_LIVE_FRAME (frame, 0);
  2849.       f = XFRAME (frame);
  2850.     }
  2851.  
  2852.   n_windows = count_windows (XWINDOW (FRAME_ROOT_WINDOW (f)));
  2853.   data = (struct save_window_data *)
  2854.            XVECTOR (Fmake_vector (make_number (SAVE_WINDOW_DATA_SIZE),
  2855.                   Qnil));
  2856.   XFASTINT (data->frame_width) = FRAME_WIDTH (f);
  2857.   XFASTINT (data->frame_height) = FRAME_HEIGHT (f);
  2858. #ifdef MULTI_FRAME
  2859.   XSET (data->selected_frame, Lisp_Frame, selected_frame);
  2860. #endif
  2861.   data->current_window = FRAME_SELECTED_WINDOW (f);
  2862.   XSET (data->current_buffer, Lisp_Buffer, current_buffer);
  2863.   data->minibuf_scroll_window = Vminibuf_scroll_window;
  2864.   data->root_window = FRAME_ROOT_WINDOW (f);
  2865.   data->focus_frame = FRAME_FOCUS_FRAME (f);
  2866.   tem = Fmake_vector (make_number (n_windows), Qnil);
  2867.   data->saved_windows = tem;
  2868.   for (i = 0; i < n_windows; i++)
  2869.     XVECTOR (tem)->contents[i]
  2870.       = Fmake_vector (make_number (SAVED_WINDOW_VECTOR_SIZE), Qnil);
  2871.   save_window_save (FRAME_ROOT_WINDOW (f),
  2872.             XVECTOR (tem), 0);
  2873.   XSET (tem, Lisp_Window_Configuration, data);
  2874.   return (tem);
  2875. }
  2876.  
  2877. DEFUN ("save-window-excursion", Fsave_window_excursion, Ssave_window_excursion,
  2878.   0, UNEVALLED, 0,
  2879.   "Execute body, preserving window sizes and contents.\n\
  2880. Restores which buffer appears in which window, where display starts,\n\
  2881. as well as the current buffer.\n\
  2882. Does not restore the value of point in current buffer.")
  2883.   (args)
  2884.      Lisp_Object args;
  2885. {
  2886.   register Lisp_Object val;
  2887.   register int count = specpdl_ptr - specpdl;
  2888.  
  2889.   record_unwind_protect (Fset_window_configuration,
  2890.              Fcurrent_window_configuration (Qnil));
  2891.   val = Fprogn (args);
  2892.   return unbind_to (count, val);
  2893. }
  2894.  
  2895. _VOID_
  2896. init_window_once ()
  2897. {
  2898. #ifdef MULTI_FRAME
  2899.   selected_frame = make_terminal_frame ();
  2900.   minibuf_window = selected_frame->minibuffer_window;
  2901.   selected_window = selected_frame->selected_window;
  2902.   last_nonminibuf_frame = selected_frame;
  2903. #else /* not MULTI_FRAME */
  2904.   extern Lisp_Object get_minibuffer ();
  2905.  
  2906.   minibuf_window = make_window ();
  2907.   FRAME_ROOT_WINDOW (selected_frame) = make_window ();
  2908.  
  2909.   XWINDOW (FRAME_ROOT_WINDOW (selected_frame))->next = minibuf_window;
  2910.   XWINDOW (minibuf_window)->prev = FRAME_ROOT_WINDOW (selected_frame);
  2911.   XWINDOW (minibuf_window)->mini_p = Qt;
  2912.  
  2913.   /* These values 9 and 10 are arbitrary,
  2914.      just so that there is "something there."
  2915.      Correct values are put in in init_xdisp */
  2916.  
  2917.   XFASTINT (XWINDOW (FRAME_ROOT_WINDOW (selected_frame))->width) = 10;
  2918.   XFASTINT (XWINDOW (minibuf_window)->width) = 10;
  2919.  
  2920.   XFASTINT (XWINDOW (FRAME_ROOT_WINDOW (selected_frame))->height) = 9;
  2921.   XFASTINT (XWINDOW (minibuf_window)->top) = 9;
  2922.   XFASTINT (XWINDOW (minibuf_window)->height) = 1;
  2923.  
  2924.   /* Prevent error in Fset_window_buffer.  */
  2925.   XWINDOW (FRAME_ROOT_WINDOW (selected_frame))->buffer = Qt;
  2926.   XWINDOW (minibuf_window)->buffer = Qt;
  2927.  
  2928.   /* Now set them up for real.  */
  2929.   Fset_window_buffer (FRAME_ROOT_WINDOW (selected_frame),
  2930.               Fcurrent_buffer ());
  2931.   Fset_window_buffer (minibuf_window, get_minibuffer (0));
  2932.  
  2933.   selected_window = FRAME_ROOT_WINDOW (selected_frame);
  2934.   /* Make sure this window seems more recently used than
  2935.      a newly-created, never-selected window.  Increment
  2936.      window_select_count so the first selection ever will get
  2937.      something newer than this.  */
  2938.   XFASTINT (XWINDOW (selected_window)->use_time) = ++window_select_count;
  2939. #endif /* not MULTI_FRAME */
  2940. }
  2941.  
  2942. _VOID_
  2943. syms_of_window ()
  2944. {
  2945.   Qwindowp = intern ("windowp");
  2946.   staticpro (&Qwindowp);
  2947.  
  2948.   Qwindow_live_p = intern ("window-live-p");
  2949.   staticpro (&Qwindow_live_p);
  2950.  
  2951. #ifndef MULTI_FRAME
  2952.   /* Make sure all windows get marked */
  2953.   staticpro (&minibuf_window);
  2954. #endif
  2955.  
  2956.   DEFVAR_LISP ("temp-buffer-show-function", &Vtemp_buffer_show_function,
  2957.     "Non-nil means call as function to display a help buffer.\n\
  2958. Used by `with-output-to-temp-buffer'.");
  2959.   Vtemp_buffer_show_function = Qnil;
  2960.  
  2961.   DEFVAR_LISP ("display-buffer-function", &Vdisplay_buffer_function,
  2962.     "If non-nil, function to call to handle `display-buffer'.\n\
  2963. It will receive two args, the buffer and a flag which if non-nil means\n\
  2964.  that the currently selected window is not acceptable.\n\
  2965. Commands such as `switch-to-buffer-other-window' and `find-file-other-window'\n\
  2966. work using this function.");
  2967.   Vdisplay_buffer_function = Qnil;
  2968.  
  2969.   DEFVAR_LISP ("minibuffer-scroll-window", &Vminibuf_scroll_window,
  2970.     "Non-nil means it is the window that C-M-v in minibuffer should scroll.");
  2971.   Vminibuf_scroll_window = Qnil;
  2972.  
  2973.   DEFVAR_LISP ("other-window-scroll-buffer", &Vother_window_scroll_buffer,
  2974.     "If non-nil, this is a buffer and \\[scroll-other-window] should scroll its window.");
  2975.   Vother_window_scroll_buffer = Qnil;
  2976.  
  2977.   DEFVAR_BOOL ("pop-up-frames", &pop_up_frames,
  2978.     "*Non-nil means `display-buffer' should make a separate frame.");
  2979.   pop_up_frames = 0;
  2980.  
  2981.   DEFVAR_LISP ("pop-up-frame-function", &Vpop_up_frame_function,
  2982.     "*If non-nil, function to call to handle automatic new frame creation.\n\
  2983. It is called with no arguments and should return a newly created frame.\n\
  2984. \n\
  2985. A typical value might be `(lambda () (new-frame pop-up-frame-alist))'\n\
  2986. where `pop-up-frame-alist' would hold the default frame parameters.");
  2987.   Vpop_up_frame_function = Qnil;
  2988.  
  2989.   DEFVAR_BOOL ("pop-up-windows", &pop_up_windows,
  2990.     "*Non-nil means display-buffer should make new windows.");
  2991.   pop_up_windows = 1;
  2992.  
  2993.   DEFVAR_INT ("next-screen-context-lines", &next_screen_context_lines,
  2994.     "*Number of lines of continuity when scrolling by screenfuls.");
  2995.   next_screen_context_lines = 2;
  2996.  
  2997.   DEFVAR_INT ("split-height-threshold", &split_height_threshold,
  2998.     "*display-buffer would prefer to split the largest window if this large.\n\
  2999. If there is only one window, it is split regardless of this value.");
  3000.   split_height_threshold = 500;
  3001.  
  3002.   DEFVAR_INT ("window-min-height", &window_min_height,
  3003.     "*Delete any window less than this tall (including its mode line).");
  3004.   window_min_height = 4;
  3005.  
  3006.   DEFVAR_INT ("window-min-width", &window_min_width,
  3007.     "*Delete any window less than this wide.");
  3008.   window_min_width = 10;
  3009.  
  3010.   defsubr (&Sselected_window);
  3011.   defsubr (&Sminibuffer_window);
  3012.   defsubr (&Swindow_minibuffer_p);
  3013.   defsubr (&Swindowp);
  3014.   defsubr (&Swindow_live_p);
  3015.   defsubr (&Spos_visible_in_window_p);
  3016.   defsubr (&Swindow_buffer);
  3017.   defsubr (&Swindow_height);
  3018.   defsubr (&Swindow_width);
  3019.   defsubr (&Swindow_hscroll);
  3020.   defsubr (&Sset_window_hscroll);
  3021.   defsubr (&Swindow_edges);
  3022.   defsubr (&Scoordinates_in_window_p);
  3023.   defsubr (&Swindow_at);
  3024.   defsubr (&Swindow_point);
  3025.   defsubr (&Swindow_start);
  3026.   defsubr (&Swindow_end);
  3027.   defsubr (&Sset_window_point);
  3028.   defsubr (&Sset_window_start);
  3029.   defsubr (&Swindow_dedicated_p);
  3030.   defsubr (&Sset_window_dedicated_p);
  3031.   defsubr (&Swindow_display_table);
  3032.   defsubr (&Sset_window_display_table);
  3033.   defsubr (&Snext_window);
  3034.   defsubr (&Sprevious_window);
  3035.   defsubr (&Sother_window);
  3036.   defsubr (&Sget_lru_window);
  3037.   defsubr (&Sget_largest_window);
  3038.   defsubr (&Sget_buffer_window);
  3039.   defsubr (&Sdelete_other_windows);
  3040.   defsubr (&Sdelete_windows_on);
  3041.   defsubr (&Sreplace_buffer_in_windows);
  3042.   defsubr (&Sdelete_window);
  3043.   defsubr (&Sset_window_buffer);
  3044.   defsubr (&Sselect_window);
  3045.   defsubr (&Sdisplay_buffer);
  3046.   defsubr (&Ssplit_window);
  3047.   defsubr (&Senlarge_window);
  3048.   defsubr (&Sshrink_window);
  3049.   defsubr (&Sscroll_up);
  3050.   defsubr (&Sscroll_down);
  3051.   defsubr (&Sscroll_left);
  3052.   defsubr (&Sscroll_right);
  3053.   defsubr (&Sscroll_other_window);
  3054.   defsubr (&Srecenter);
  3055.   defsubr (&Smove_to_window_line);
  3056.   defsubr (&Swindow_configuration_p);
  3057.   defsubr (&Sset_window_configuration);
  3058.   defsubr (&Scurrent_window_configuration);
  3059.   defsubr (&Ssave_window_excursion);
  3060. }
  3061.  
  3062. _VOID_
  3063. keys_of_window ()
  3064. {
  3065.   initial_define_key (control_x_map, '1', "delete-other-windows");
  3066.   initial_define_key (control_x_map, '2', "split-window");
  3067.   initial_define_key (control_x_map, '0', "delete-window");
  3068.   initial_define_key (control_x_map, 'o', "other-window");
  3069.   initial_define_key (control_x_map, '^', "enlarge-window");
  3070.   initial_define_key (control_x_map, '<', "scroll-left");
  3071.   initial_define_key (control_x_map, '>', "scroll-right");
  3072.  
  3073.   initial_define_key (global_map, Ctl ('V'), "scroll-up");
  3074.   initial_define_key (meta_map, Ctl ('V'), "scroll-other-window");
  3075.   initial_define_key (meta_map, 'v', "scroll-down");
  3076.  
  3077.   initial_define_key (global_map, Ctl('L'), "recenter");
  3078.   initial_define_key (meta_map, 'r', "move-to-window-line");
  3079. }
  3080.